৩-৩ঃ কালেকশনস (অ্যারে ও ডিকশনারী), ইনুম্যারেশন ও ক্লোজার

Standard

এই সিরিজের পুরো পোস্ট লিস্ট এবং সম্ভাব্য কাগুজে বই এর সূচি দেখতে ক্লিক করুনbangla-ios-objective-c-swift-nuhil

আগের চ্যাপ্টারঃ সুইফ্ট ল্যাঙ্গুয়েজে স্ট্রিং ও ক্যারেকটার টাইপ ভ্যারিয়েবল

ভুমিকাঃ
অ্যাপলের নতুন প্রোগ্রামিং ল্যাঙ্গুয়েজ সুইফ্ট নিয়ে লেখা আমাদের তৃতীয় সেকশনের প্রথম অধ্যায়ে সুইফ্ট সম্পর্কে পরিচিতি মুলক আলোচনা হয়েছে। আমরা সুইফ্ট ল্যাঙ্গুয়েজের বেসিক সিনট্যাক্স, ভ্যারিয়েবল ও কনস্ট্যান্ট ডিক্লেয়ার করা, বিভিন্ন ধরনের অপারেটর সম্পর্কে জেনেছি। দ্বিতীয় অধ্যায়ে স্ট্রিং ও ক্যারেকটার নিয়ে বিস্তারিত আলোচনা হয়েছে। এই অধ্যায়ে অ্যারে, ডিকশনারী ইত্যাদি কালেকশনস (Collections) ও অল্পবিস্তর ইনুম্যারেশন ও ক্লোজার নিয়ে আলোচনা হবে।

কালেকশন টাইপঃ
ধরুন, আপনি একজন রেস্টুরেন্টের মালিক। আপনি আপনার রেগুলার কাস্টমারদের একটি তালিকা করতে চাচ্ছেন সুইফ্ট ল্যাঙ্গুয়েজে। একটু সহজ করার জন্য ধরে নিচ্ছি যে আপনি শুধু কাস্টমারদের নামের তালিকা করবেন। ভাবুন তো কিভাবে করবেন। যদি আপনার ১০০ জন কাস্টমার থাকে তাহলে ১০০ টা ভ্যারিয়েবল ডিক্লেয়ার করবেন। প্রত্যেকটিতে একজন করে কাস্টমারের নাম স্টোর করবেন।

var 1stCust = "First Customer's Name"
var 2ndCust = "2nd Customer's Name"
var 3rdCust = "3rd Customer's Name"
.
.
var 100thCust = "100th Customer's Name"

আচ্ছা, এবার বলুন তো আপনি যদি এই লিস্টটা দেখতে চান তাহলে কি করবেন?

println(1stCust)
println(2ndCust)
.
.
println(100thCust)

এভাবে করার সমস্যাটা হলঃ

  • আপনি নিশ্চয়ই জানেন না যে আপনার কাস্টমার সংখ্যা কত হতে পারে।
  • এভাবে ধরে ধরে এক-একটি ভ্যারিয়েবলে ডাটা স্টোর করতে হবে। আবার এক-একটি ধরে ডাটা ডিসপ্লে করতে হবে।
  • কোন রকমের সার্চিং টেকনিক (Searching Technique) অ্যাপ্লাই করা যাবে না। অর্থাৎ কোন নির্দিস্ট কাস্টমারের নাম খুজে বের করা যাবে না।
  • কোন রকমের সর্টিং টেকনিক (Sorting Technique) অ্যাপ্লাই করা যাবে না। অর্থাৎ লিস্টটিকে কোন নির্দিস্ট অর্ডারে সাজানো যাবে না।

এসব সমস্যার সমাধানের জন্য সুইফ্টে রয়েছে দুই ধরনের কালেকশন টাইপ (Collection Type) যথাঃ অ্যারে (Arrays) ও ডিকশনারী (Dictionaries)।

অ্যারে ও ডিকশনারী (Arrays and Dictionaries)ঃ
একই ডাটাটাইপের অনেকগুলো ভ্যারিয়েবলের কালেকশনকেই অ্যারে অথবা ডিকশনারী বলা হয়। অ্যারে ও ডিকশনারী উভয়ই একটি নির্দিষ্ট ডাটাটাইপের এক বা একাধিক ডাটা সংরক্ষন করতে পারে। অ্যারেতে ভ্যালুগুলো সবসময় একটি অর্ডারে সাজানো থাকে। 0 থেকে শুরু করে 1, 2, 3, …, n-1 পর্যন্ত অর্ডারে ডাটাগুলো থাকে। এই ০ থেকে n-1 কে অ্যারের ইনডেক্স বলা হয়। এখানে n মানে হল অ্যারের সাইজ বা অ্যারেতে থাকা ভ্যালুর সংখ্যা। এই ইনডেক্স ব্যবহার করেই অ্যারের ভ্যালু গুলো ম্যানিপুলেট করা হয়। অন্যদিকে ডিকশনারীতে ভ্যালুগুলো কোন নির্দিষ্ট অর্ডারে সাজানো থাকে না। এখানে ইনডেক্সকে বলা হয় কি (Key)। key হল নির্দিষ্ট ইউনিক আইডেন্টিফায়ার যা দিয়ে ডিকশনারীর ভ্যালুগুলোকে সংরক্ষন বা এক্সেস করা যায়। যেমন ঃ

restaurantArr[0] = 10; // Array
restaurantArr[1] = 15; // Array
restaurantDict["numberOfItem"] = 10  // Dictionary

অ্যারে (Array)
একটি অ্যারেতে একই ডাটাটাইপের একাধিক ভ্যালু থাকতে পারে। ০ থেকে শুরু করে ১, ২, ৩, … অর্ডারে সাজানো থাকে ভ্যালুগুলো। আবার একই ভ্যালু একাধিক বারও থাকতে পারে। অবজেকটিভ-সি এর NSArray এবং NSMutableArray তে যেকোন অবজেক্ট ভ্যালু হিসেবে রাখা যায়। এমনকি একটি অ্যারে তে বিভিন্ন টাইপের ভ্যালু বা অবজেক্ট থাকতে পারে। কিন্তু সুইফ্ট -এ অ্যারেতে একটি নির্দিষ্ট টাইপের ডাটা রাখা যাবে। উদাহরনস্বরুপ Int টাইপের অ্যারেতে শুধু Int টাইপের ডাটাই রাখা যাবে। অন্য কিছু না।

অ্যারে টাইপ ভ্যারিয়েবল ডিক্লেয়ার করাঃ
বিভিন্নভাবে অ্যারে টাইপ ভ্যারিয়েবল ডিক্লেয়ার বা ইনিশিয়ালাইজ করা যায়। নিচে দুই ধরনের ডিক্লেয়ারেশন এর উদাহরন দেওয়া হলঃ

 Array<Int>variableA
 var variableB: Int[] = [1, 2, 4]

উপরের কোডটিতে variableA এবং variableB নামের দুটি অ্যারে ডিক্লেয়ার করা হয়েছে। দুই ভাবেই অ্যারে ডিক্লেয়ার করা যায়। প্রথমটিকে ফুল ফর্ম আর দ্বিতীয়টিকে শর্টহ্যান্ড সিনট্যাক্স বলা হয়। শর্টহ্যান্ড সিনট্যাক্সই সবচেয়ে বেশী ব্যবহার করা হয়। এক্ষেত্রে var অথবা let লিখে তারপর অ্যারের নাম লিখতে হয়। এরপর কোলন (:) দিয়ে ডাটাটাইপ লিখে তারপর বন্ধনী ( [] ) চিহ্ন দিতে হয়।

অ্যারেতে সরাসরি [value1, value2,…..,valueN] অ্যাসাইন করে অ্যারে ইনিশিয়ালাইজ করা যায়। যেমনঃ

 var variableInt: Int[] = [12, 12, 34]
//variableInt is initialised with 12,12 and 34

 var variableString = ["Egg", "Fruits"] 
// variableString is initialised with Egg and Fruits with these two string

সুইফ্ট এর টাইপ ইনফারেন্স (Type Inference) সুবিধার জন্য অ্যারে ডিক্লেয়ার বা ইনিশিয়ালাইজেশনের সময় অ্যারের টাইপ না লিখলেও অ্যাসাইন করা ভ্যালুর টাইপই হবে অ্যারে টাইপ। তাই variableString একটি String টাইপ অ্যারে কারন দুটি String টাইপের ভ্যালু দিয়ে অ্যারেটি ইনিশিয়ালাইজ করা হয়েছে।

অ্যারে ম্যানিপুলেশন (এক্সেস ও মডিফিকেশন)
অ্যারের ভ্যালুগুলো পড়া (Access) বা আপডেট (Modification) করার জন্য সুইফ্ট ল্যাঙ্গুয়েজের অ্যারে টাইপ ভ্যারিয়েবলের জন্য কিছু প্রোপার্টি, মেথড আছে। এসব প্রোপার্টি বা মেথড দিয়ে অথবা সাবস্ক্রিপ্ট দিয়ে অ্যারে এক্সেস বা মডিফাই করা যায়। নিচে কিছু উদাহরন দিয়ে বিস্তারিত বর্ননা করা হলঃ
কোন অ্যারেতে কতগুলো ভ্যালু আছে তা জানা যাবে রিড-অনলি প্রোপার্টি count দিয়ে।

var menuItems = ["Eggs", "Potatoes", Chilis"]
println("menuItems array has \(menuItems.count) items.") // menuItems array has 3 items. 

প্রথমে menuItems অ্যারেটি ৩ টি স্ট্রিং টাইপ ভ্যালু দিয়ে ইনিশিয়ালাইজ করা হয়েছে। তারপর menuItems.count প্রোপার্টিতে অ্যারেতে কয়টি ভ্যালু আছে তা রিটার্ন করে। তাই menuItems.count ৩ রিটার্ন করবে।

বুলিয়ান isEmpty প্রোপার্টি দিয়ে কোন অ্যারে ফাঁকা কিনা তা চেক করা যায়। অর্থাৎ যদি কোন অ্যারের count ০ হয় বা অ্যারেতে কোন ইলিমেন্ট না থাকে তাহলে isEmpty প্রোপার্টি YES (true) রিটার্ন করে।

if menuItems.isEmpty {
    println("The menu list is empty.")
} else {
    println("The menu list is not empty.")
}
// prints "The menu list is not empty."

যেহেতু menuItems অ্যারেতে ৩টি স্ট্রিং ইলিমেন্ট আছে তাই menuItems.isEmpty প্রোপার্টি NO রিটার্ন করবে।

অ্যারের append মেথড ও += অপারেটর দিয়ে অ্যারেতে নতুন ভ্যালু যোগ করা যায়। আবার সরাসরি একটি নতুন অ্যারে কে += অপারেটর দিয়ে অ্যারের শেষে যোগ করা যায়। নিচে উদাহরন দেওয়া হলঃ

menuItems.append("Oils")
//Oils is appended on the menuItems array. Now menuItems has 4 strings.

menuItems += "Flour" 
// Flour is appended on the menuItems array. Now menuItems has 5 strings

menuItems += ["Baking powder", "Butter", "Chocolate" ]
//An array of 3 new items is appended on the menuItems array. Now menuItems has 8 strings.

আগেই বলেছি অ্যারেতে ভ্যালুগুলো ০, ১, ২, … অর্ডারে সাজানো থাকে। যেমন অ্যারের ৫ নম্বর ভ্যালু তথা ৫-১ বা ৪ তম ইনডেক্স এর ভ্যালু এক্সেস করার জন্য সাবস্ক্রিপ্ট সিনট্যাক্স ব্যবহার করে নিচের মত স্কয়ার ব্র্যাকেটের মধ্যে ইনডেক্স পাঠাতে হয়।

var 5thItem = menuItems[4]

এক্ষেত্রে অবশ্যই খেয়াল রাখতে হবে যাতে পাঠানো ইনডেক্সটি অ্যারেতে থাকা ভ্যালুর সংখ্যার সর্বোচ্চ ইনডেক্সের বেশী না হয়। ধরুন অ্যারেতে ০ থেকে ৪ ইনডেক্সে ৫ টি ভ্যালু রয়েছে। কিন্তু আপনি ৫ নম্বর ইনডেক্স তথা ৬ষ্ঠ ভ্যালু চাচ্ছেন। সেক্ষেত্রে “Array out of bound” টাইপের এরর হতে পারে।

একই ভাবে সাবস্ক্রিপ্ট সিনট্যাক্স দিয়ে কোন নির্দিষ্ট ইনডেক্স বা কোন রেন্জের মধ্যকার সকল ইনডেক্সের ভ্যালু আপডেট বা মডিফিকেশন করা যায়।

menuItems[4] = "Onion" // "Flour" is replaced by "Onion"
menuItems[1..3] = ["Bananas", "Apples"]
// Potatoes and Chilis are replaced by Bananas and Apples

এখানে, দ্বিতীয় উদাহরনটিতে 1..3 মানে 1 থেকে 3 রেন্জের (শেষ ইনডেক্স, এক্ষেত্রে ৩য় ইনডেক্স কন্সিডার হয় না) সকল ইনডেক্সের ভ্যালুকে একটি অ্যারে দিয়ে রিপ্লেস করা হচ্ছে যাতে দুটি ভ্যালু রয়েছে। অর্থাৎ সাবস্ক্রিপ্ট সিনট্যাক্স ব্যবহার করে যেকোন সংখ্যক ইনডেক্সের ভ্যালু অন্য যেকোন সংখ্যক ভ্যালু দিয়ে রিপ্লেস করা যাবে।
উল্লেখযোগ্য কথা হল সাবস্ক্রিপ্ট সিনট্যাক্স (স্কয়ার ব্র্যাকেটের ভিতর অ্যারে ইনডেক্স পাঠানো) ব্যবহার করে অ্যারের ভ্যালু রিড করা বা নতুন ভ্যালু দিয়ে রিপ্লেস করা যায় ঠিকই কিন্তু নতুন কোন ভ্যালু অ্যারের শেষে ইনসার্ট (Add/ Append/ Insert) করা যায় না।

var recipes = ["Chicken Resala", "Mutton Curry"]
recipes[2] = "another recipe"   // Throws run time error because index 2 doesn't have value
println(recipes[2])      // Throws run time error because index 2 doesn't have value

recipes অ্যারেতে ০, ১ ইনডেক্সে মোট দুটি ভ্যালু রয়েছে। যদি ইনডেক্স ২ এর ভ্যালু এক্সেস করার বা নতুন ভ্যালু সেট করার চেষ্টা করা হয় তাহলে রান টাইম এরর হয়। অর্থাৎ সাকসেসফুল কম্পাইলেশনের পর কোড রান করলে যখনই মেশিন recipes অ্যারের ২ নম্বর ইনডেক্সের ভ্যালু এক্সেস করতে চাইবে তখনি এরর হবে কারন এই অ্যারেতে ২ নম্বর ইনডেক্স নেই।

কোন নির্দিষ্ট ইনডেক্সে নতুন ভ্যালু ইনসার্ট করার জন্য রয়েছে অ্যারের Insert মেথড। Insert মেথডে দুটি আরগুমেন্ট পাঠানো হয়। প্রথমে নতুন ভ্যালু ও তারপর যে ইনডেক্সে ইনসার্ট করতে হবে তার ভ্যালু।

menuItems.insert("Maple Syrup", atIndex: 2)

menuItems অ্যারের ২ নম্বর ইনডেক্সে Maple Syrup ইনসার্ট হয়ে গেছে। পরবর্তী সকল ভ্যালুর ইনডেক্স ১ করে বেড়ে গেছে। অর্থাৎ এই ইনসার্ট অপারেশন চালানোর পুর্বে যে ভ্যালুটি ইনডেক্স ২ তে ছিল এখন সেটি ইনডেক্স ৩ এ আছে।

নিচের মত করে removeAtIndex() মেথড দিয়ে অ্যারের যেকোন ইনডেক্সের ভ্যালু রিমুভ করা যায়। সেক্ষেত্রে রিমুভ করার পর ওই ইনডেক্সের পরবর্তী সকল ভ্যালুর ইনডেক্স ১ করে কমে যাবে।

menuItems.removeAtIndex(2)

ইনডেক্স ২ এর ভ্যালু রিমুভ হয়ে পরবর্তীতে থাকা সকল ভ্যালুর ইনডেক্স ১ করে কমে যাবে।

removeLast() মেথড দিয়ে সহজেই যেকোন অ্যারের শেষ ভ্যালুটি রিমুভ করা যায়। ফলে অ্যারের count ও ১ কমে যায়।

menuItems.removeLast() // removes the last element

for-in লুপ ব্যবহার করে অ্যারের সবগুলো ইনডেক্স ইটেরেট করা যায়। নিচের উদাহরনটি দেখা যাকঃ

 for item in menuItems
 {   
    println(item)
 }

অ্যারে ইনিশিয়ালাইজেশনঃ
শুরুতে কোন ভ্যালু ছাড়াই ফাঁকা অ্যারে ডিক্লেয়ার করার জন্য অ্যারের ডাটাটাইপ লিখে তারপর স্কয়ার ব্র্যাকেট লিখতে হয় নিচের মত করে।

var someInts = Int[]()
println("someInts is of type Int[] with \(someInts.count) items.")

সুইফ্ট এর Array তে রয়েছে এমন একটি ইনিশিয়ালাইজার যা দিয়ে একটি নির্দিষ্ট সাইজের অ্যারে ডিক্লেয়ার করা যাবে এবং প্রতিটি ইনডেক্সে একটি ডিফল্ট ভ্যালু সেট করা যাবে। উদাহরনস্বরুপ ধরুন আপনি চাচ্ছেন এমন একটি অ্যারে ডিক্লেয়ার করতে যার সাইজ (count) হবে ১০০ (ইনডেক্স 0 – 99) এবং প্রতিটি ইনডেক্সের ডিফল্ট ভ্যালু হবে 0.0 (repeatedValue)।

var hundredDoubles = Double[](count: 100, repeatedValue: 0.0)
// hundredDoubles is of type Double[], and equals [0.0, 0.0, 0.0, ...]

hundredDoubles অ্যারেটি ডিক্লেয়ার করার সময় ব্র্যাকেটের মধ্যে দুটি আর্গুমেন্ট পাঠানো হয়। প্রথমটি count যা অ্যারের সাইজ কত হবে তা ডিফাইন করে। দ্বিতীয়টি repeatedValue যা অ্যারের সকল ইনডেক্সের ডিফল্ট ভ্যালু কত হবে তা সেট করে। আবার Double[] লিখে অ্যারের ভ্যালুগুলোর ডাটাটাইপ কি হবে তাও বলে দেওয়া হয়েছে।
সুইফ্ট এর টাইপ ইনফারেন্স সুবিধার জন্য ডাটাটাইপ না বলে দিয়ে নিচের মত করে অ্যারে ইনিশিয়ালাইজ করা যায়।

var anotherThreeDoubles = Array(count: 3, repeatedValue: 2.5)
// anotherThreeDoubles is inferred as Double[], and equals [2.5, 2.5, 2.5]

একথা বলাই বাহুল্য যে দুটি অ্যারে কে “+” অপারেটর দিয়ে যোগ করলে প্রথম অ্যারের শেষে দ্বিতীয় অ্যারেটি অ্যাপেন্ড হয়।

var sixDoubles = threeDoubles + anotherThreeDoubles
// sixDoubles is inferred as Double[], and equals [0.0, 0.0, 0.0, 2.5, 2.5, 2.5]

ডিকশনারী (Dictionaries) ঃ
অ্যারের মতই ডিকশনারীতে একই টাইপের একাধিক ভ্যালু স্টোর করা যায়। প্রত্যেকটি ভ্যালুর জন্য একটি করে ইউনিক কি (key) বা আইডেন্টিফায়ার থাকে। অ্যারেতে ০ থেকে শুরু করে ১, ২, ৩, … ইনডেক্স গুলোতে ভ্যালু গুলো থাকে কিন্তু ডিকশনারীতে ভ্যালুগুলো অ্যারের মত ইনডেক্সিং করে সাজানো থাকে না। ঠিক বাস্তব ডিকশনারী তে যেমন নির্দিষ্ট শব্দের জন্য নির্দিষ্ট সংজ্ঞা থাকে তেমনি সুইফ্ট এর ডিকশনারীতে প্রত্যেকটি ভ্যালু নিজের ইউনিক আইডেন্টিফায়ার দিয়ে স্টোর করা থাকে। এই আইডেন্টিফায়ার বা ভ্যালু উভয়টি যেকোন টাইপের হতে পারে। তবে একটি ডিকশনারীর সকল ভ্যালুর টাইপ একই হবে এবং সকল আইডেন্টিফায়ারের টাইপ একই হবে।

অবজেকটিভ-সি এর NSDictionary ও NSMutableDictionary তে একই ডিকশনারীতে বিভিন্ন টাইপের ভ্যালু বা আইডেন্টিফায়ার (key) থাকতে পারে। ফলে ডিকশনারীর টাইপ নির্দিষ্ট থাকে না। অন্যদিকে যেহেতু সুইফ্টের ডিকশনারীর সকল ইলিমেন্ট এর টাইপ একই হয় তাই ডিকশনারীর ভ্যালুর টাইপ সম্পর্কে নিশ্চিন্ত থাকা যায়।

ডিকশনারী (Dictionaries) ডিক্লেয়ারেশন ঃ
সুইফ্টে ডিকশনারী টাইপ ডিক্লেয়ার করার জন্য Dictionary লিখতে হয়। এখানে KeyType হল সেই আইডেন্টিফায়ার বা কি (key) এর টাইপ এবং ValueType হল ডিকশনারীতে স্টোর করা ভ্যালুর টাইপ। যেমনঃ

var airports: Dictionary<String, String> = ["TYO": "Tokyo", "DUB": "Dublin"]

ডিকশনারী ইনিশিয়ালাইজ করার জন্য অ্যারের মতই শর্টহ্যান্ড সিনট্যাক্স ব্যবহার করা হয়। অর্থাৎ ভ্যারিয়েবল ডিক্লেয়ার করার সময় স্কয়ার ব্র্যাকেটের ভিতর ভ্যালুগুলো লিখে দেওয়া। কিন্তু ডিকশনারী তে যেহেতু অ্যারের মত ইনডেক্সিং হয় না, তাই এখানে কি-ভ্যালু পেয়ার (key-value pairs) দিয়ে ডিকশনারী ইনিশিয়ালাইজ করতে হয়। উপরের উদাহরনটিতে ডিকশনারীর ইলিমেন্টগুলো কমা (,) দিয়ে লেখা হয়েছে। এবং প্রত্যেক ইলিমেন্ট এর কোলন (:) দিয়ে কি-ভ্যালু পেয়ার লেখা হয়েছে। এখানে কোলনের ডানদিকের অংশটি হল ভ্যালু ও বামদিকের অংশটি হল এই ভ্যালুর জন্য ইউনিক আইডেন্টিফায়ার বা কি (key)।

নোট ঃ আমরা বার বার বলছি যে ডিকশনারীর কি (key) গুলো অবশ্যই ইউনিক হতে হবে। তা না হলে একই কি(key) এর জন্য যদি একাধিক ভ্যালু স্টোর করা হয় তাহলে তা Ambiguity তৈরী করবে। এজন্যই সুইফ্টে কি(key) এর জন্য এমন ডাটাটাইপ ব্যবহার করতে হবে যেগুলোকে হ্যাশ করা যায়। String, Int, Double, Bool ইত্যাদি ডাটাটাইপ বাই-ডিফল্ট হ্যাশ করা যায়। তাই এসবগুলো ডাটাটাইপই ডিকশনারীর কি(key) হিসেবে ব্যবহার করা যাবে।

ডিকশনারী ম্যানিপুলেশন (এক্সেস ও মডিফিকেশন)ঃ
ডিকশনারীর ভ্যালুগুলো পড়া (Access) বা আপডেট (Modification) করার জন্য সুইফ্ট ল্যাঙ্গুয়েজের ডিকশনারী টাইপ ভ্যারিয়েবলের জন্য আছে কিছু প্রোপার্টি, মেথড। এসব প্রোপার্টি বা মেথড দিয়ে অথবা সাবস্ক্রিপ্ট দিয়ে ডিকশনারী এক্সেস বা মডিফাই করা যায়। নিচে কিছু উদাহরন দিয়ে বিস্তারিত বর্ননা করা হলঃ

কোন ডিকশনারীতে কতগুলো ভ্যালু আছে তা জানা যাবে রিড-অনলি প্রোপার্টি count দিয়ে।

var airports: Dictionary<String, String> = ["TYO": "Tokyo", "DUB": "Dublin"]
println("airports array has \(menuItems.count) items.") // airports dictionary has 2 items. 

প্রথমে airports ডিকশনারীটি 2 টি স্ট্রিং টাইপ ভ্যালু দিয়ে ইনিশিয়ালাইজ করা হয়েছে যাদের কি-ভ্যালু পেয়ার হল String : String। তারপর airports.count প্রোপার্টিতে ডিকশনারীতে কয়টি ভ্যালু আছে তা রিটার্ন করে। তাই airports.count 2 রিটার্ন করবে।

সাবস্ক্রিপ্ট সিনট্যাক্স দিয়ে সহজেই ডিকশনারীতে নতুন ভ্যালু ইনসার্ট বা আগের কোন ভ্যালু আপডেট করা হয়। সাবস্ক্রিপ্ট এ ইনডেক্স হিসেবে কি (key) লিখে তাতে নতুন ভ্যালু অ্যাসাইন করলে প্রথমেই চেক করা হয় যে এই কি (key) দিয়ে কোন ভ্যালু ডিকশনারীতে আছে কিনা। যদি থাকে তাহলে নতুন ভ্যালু দিয়ে আপডেট হবে। আর যদি না থাকে তাহলে নতুন ইলিমেন্ট টি ডিকশনারীতে ইনসার্ট হবে।

airports["LHR"] = "London"  // new item with "LHR" : "London" key-pair is inserted
// the airports dictionary now contains 3 items

airports["LHR"] = "London Heathrow" // existing item of key "LHR" is updated by ""London Heathrow""
// the value for "LHR" has been changed to "London Heathrow"

সাবস্ক্রিপ্ট সিনট্যাক্স ব্যবহার না করে ডিকশনারীর updateValue(forKey:) মেথড দিয়ে কোন নির্দিষ্ট কি (key) এর ভ্যালু আপডেট করা যায় আবার নতুন কি-পেয়ার ভ্যালু ইনসার্ট করা যায়।

if let oldValue = airports.updateValue("Dublin International", forKey: "DUB") {
    println("The old value for DUB was \(oldValue).")
}
// prints "The old value for DUB was Dublin."

উপরের কোডটিতে দেখা যাচ্ছে updateValue মেথডে দুটি আর্গুমেন্ট পাঠানো হচ্ছে। প্রথমটিতে ভ্যালু এবং দ্বিতীয়টিতে কি(key)। এখানে ভ্যালু হিসেবে “Dublin International” এবং forKey:”DUB” এ কি (key) হিসেবে “DUB” পাঠানো হচ্ছে। প্রথমে চেক করা হচ্ছে যে DUB কি দিয়ে কোন ইলিমেন্ট ডিকশনারীতে আছে কিনা। যেহেতু airports ডিকশনারীতে এই কি দিয়ে একটি ভ্যালু আছে তাই এই কি এর পুর্বের ভ্যালুটি নতুন পাঠানো Dublin International দিয়ে আপডেট হয়ে গেছে এবং পুর্বের ভ্যালুটি রিটার্ন করে যা oldValue ভ্যারিয়েবলে অ্যাসাইন হয়। যদি airports ডিকশনারীতে এই কি দিয়ে কোন ভ্যালু না থাকত, তাহলে নতুন ইলিমেন্ট হিসেবে “DUB” : “Dublin International” কি-পেয়ার ডিকশনারীতে যুক্ত হত।

সাবস্ক্রিপ্ট সিনট্যাক্সে স্কয়ার ব্র্যাকেটের মধ্যে কি(key) লিখে ওই কি এর ভ্যালু রিড বা এক্সেস করা যায়।

var airport = airports["DUB"]
println(airport) // Dublin International

কোন নির্দিষ্ট কি এর ভ্যালু রিমুভ করতে চাইলে তার সাবস্ক্রিপ্ট এ nil অ্যাসাইন করলেই তা ডিকশনারী থেকে রিমুভ হয়ে যাবে।

  airports["DHK"] = "Shahjalal International Airport" 
// airports contains 3 items right now.

  airports["DHK"] = nil
// Value for "DHK" key is removed from airports and this dictionary contains two items

আবার সাবস্ক্রিপ্ট ব্যবহার না করে removeValueForKey() মেথড দিয়ে কোন নির্দিষ্ট কি-ভ্যালু পেয়ার ডিকশনারী থেকে রিমুভ করা যায়। এক্ষেত্রে মেথডটি রিমুভ হওয়া ভ্যালুটি রিটার্ন করে।

if let removedValue = airports.removeValueForKey("DUB") {
    println("The removed airport's name is \(removedValue).")
} else {
    println("The airports dictionary does not contain a value for DUB.")
}
// prints "The removed airport's name is Dublin International."

অ্যারের মতই for-in লুপ দিয়ে ডিকশনারীতে থাকা সকল ভ্যালুতে ইটেরেট করা যায়। ডিকশনারী তে কি-ভ্যালু পেয়ার হিসেবে ভ্যালু থাকে তাই এখানে অ্যারের মতই একই সাথে কি এবং ভ্যালু এক্সেস করা যায়।

for (airportCode, airportName) in airports {
    println("\(airportCode): \(airportName)")
}
// TYO: Tokyo
// LHR: London Heathrow

আবার চাইলে শুধু keys অথবা শুধু values প্রোপার্টি ইটেরেট করে শুধু কি বা শুধু ভ্যালুগুলো এক্সেস করা যায়।

for airportCode in airports.keys {
    println("Airport code: \(airportCode)")
}
// Airport code: TYO
// Airport code: LHR
 
for airportName in airports.values {
    println("Airport name: \(airportName)")
}
// Airport name: Tokyo
// Airport name: London Heathrow

একটি ফাঁকা ডিকশনারী (Empty Dictionary) ইনিশিয়ালাইজ করা ঃ
ইনিশিয়ালাইজার সিনট্যাক্স ব্যবহার করে অ্যারের মতই ফাকা ডিকশনারী ইনিশিয়ালাইজ করা যায়।

var namesOfIntegers = Dictionary<Int, String>()
// namesOfIntegers is an empty Dictionary<Int, String>

উদাহরনটিতে nameofIntegers নামের একটি ডিকশনারী ইনিশিয়ালাইজ করা হয়েছে যার কি গুলো Integer টাইপ, কিন্তু ভ্যালুগুলো হবে String টাইপ।

কোন ডিকশনারী ইনিশিয়ালাইজ করার সময় যদি টাইপ বলে দেওয়া হয় অথবা কোন ভ্যালু ইনিশিয়ালাইজ করার ফলে টাইপ ইনফারেন্স ঘটে তাহলে ডিকশনারী টি ফাঁকা করলেও টাইপ পরিবর্তন হয় না। এই বৈশিষ্ট্য অ্যারের ক্ষেত্রেও প্রযোজ্য।

namesOfIntegers[16] = "sixteen"
// namesOfIntegers now contains 1 key-value pair
namesOfIntegers = [:]
// namesOfIntegers is once again an empty dictionary of type Int, String

কোন ডিকশনারী তে [:] এভাবে স্কয়ার ব্র্যাকেটের ভিতর শুধু কোলন (:) দিয়ে অ্যাসাইন করলে তা ফাকা ডিকশনারীতে পরিনত হয়।

ইনুম্যারেশন ও ক্লোজার (Enumerations and Closures) ঃ
ইনুম্যারেশন দিয়ে একই ধরনের একগুচ্ছ ভ্যালু টাইপকে প্রকাশ করা হয়। আপনি যদি সি/সি++ এর সাথে পরিচিত থাকেন তাহলে নিশ্চয়ই জানেন যে, সি তে একটি এনাম (enum) টাইপ গ্রুপের সকল ভ্যারিয়েবলকে ইনটিজার(Integer) ভ্যালু দিয়ে অ্যাসাইন করা হয়। সুইফ্ট এ ইনুম্যারেশন অনেক বেশী ফ্লেক্সিবল এবং সি এর মত গ্রুপের ভ্যালুগুলো শুধু ইনটিজার ভ্যালু হয় না। যেকোন ডাটাটাইপ যেমন String, Character, Integer বা Float টাইপ হতে পারে।

enum CompassPoint {
    case North
    case South
    case East
    case West
}

সব ল্যাঙ্গুয়েজেই enum কিওয়ার্ড দিয়ে ইনুম্যারেশন টাইপ ডিক্লেয়ার করা হয়। উপরের উদাহরনটিতে CompassPoint নামে একটি enum টাইপ ডিক্লেয়ার করা হয়েছে যার চারটি সদস্য (member) যথাক্রমে North, South, East এবং West রয়েছে। সি/সি++ বা অন্য যেকোন ল্যাঙ্গুয়েজে enum টাইপের প্রত্যেকটি সদস্য (member) এ ০ থেকে ১, ২, ৩, … integer ভ্যালু অ্যাসাইন হয়। কিন্তু সুইফ্ট এর enum টাইপ এর member দের ভ্যালুতে এরকম কোন ডিফল্ট ভ্যালু অ্যাসাইন করা হয় না।

কোন নির্দিষ্ট ফাংশনালিটি সহ একটি কোড ব্লককে ক্লোজার (Closure) বলা হয় ({ })। সুইফ্টের ক্লোজার ঠিক অবজেকটিভ-সি এর ব্লক (Block) এর মতই কাজ করে। নিচে ক্লোজারের একটি জেনারেল ফর্ম লেখা হল।

{ (param: Type) -> ReturnType in
    expression_using_params
}

উপরের “->” চিহ্নটি আর্গুমেন্ট ও রিটার্ন টাইপকে আলাদা করে এবং “in” ক্লোজার এর হেডার থেকে ক্লোজার এর বডি আলাদা করে। নিচে একটি উদাহরণ,

var numbers = [1, 2, 3, 4, 5]
numbers.map({
    (number: Int) -> Int in
    let result = 3 * number
    return result
    })

যদি টাইপ জানা থাকে (যেমন উপরে), তাহলে নিচের মত করেও এই ক্লোজারটি ব্যবহার করা যেতে পারে,

var numbers = [1, 2, 6]
numbers = numbers.map({ number in 3 * number })
println(numbers) // [3, 6, 18]

পরিসমাপ্তিঃ
এ অধ্যায়ে কালেকশন টাইপ নিয়ে বিস্তারিত আলোচনা হয়েছে। আমরা দেখেছি কিভাবে অ্যারে ও ডিকশনারী ইনিশিয়ালাইজ করা ও ম্যানিপুলেট করতে হয়। এছাড়া জেনেছি অ্যারে ও ডিকশনারীর মধ্যকার পার্থক্য ও সুবিধা-অসুবিধা। এছাড়া ইনুম্যারেশন ও ক্লোজার সম্পর্কে নামমাত্র আলোচনা হয়েছে। এ অধ্যায় সম্পর্কিত আরও বিস্তারিত থাকবে আমাদের কাগুজে বইয়ে।

বইয়ের আপডেট পেতে চোখ রাখুন আমাদের ফ্যান পেজে

পরের চাপ্টারঃ 
পরের চাপ্টারে Branching ও Lopping নিয়ে বিস্তারিত থাকবে। ওই অধ্যায়ে if-else, switch, for-in ইত্যাদি নিয়ে বিস্তারিত আলোচনা থাকবে।

৩-২ঃ সুইফ্ট ল্যাঙ্গুয়েজে স্ট্রিং ও ক্যারেকটার টাইপ ভ্যারিয়েবল

Standard

এই সিরিজের পুরো পোস্ট লিস্ট এবং সম্ভাব্য কাগুজে বই এর সূচি দেখতে ক্লিক করুনbangla-ios-objective-c-swift-nuhil

আগের চ্যাপ্টারঃ সুইফ্ট (Swift) – অ্যাপলের নতুন চমক (পরিচিতি ও অন্যান্য বেসিক)

ভুমিকাঃ
অ্যাপলের নতুন প্রোগ্রামিং ল্যাঙ্গুয়েজ সুইফ্ট নিয়ে লেখা আমাদের তৃতীয় সেকশনের প্রথম অধ্যায়ে সুইফ্ট সম্পর্কে পরিচিতি মুলক আলোচনা হয়েছে। আমরা সুইফ্ট ল্যাঙ্গুয়েজের বেসিক সিনট্যাক্স, ভ্যারিয়েবল ও কনস্ট্যান্ট ডিক্লেয়ার করা, বিভিন্ন ধরনের অপারেটর সম্পর্কে জেনেছি। এই অধ্যায়ে ক্যারেকটার ও স্ট্রিং ম্যানিপুলেশনের বিস্তারিত থাকবে।

ক্যারেকটার ও স্ট্রিং ঃ
অন্যান্য বেসিক ডাটাটাইপ (Integer, Float, Double) গুলোর মতই ক্যারেকটার (Character) একটি ডাটাটাইপ যা এক বাইট (Byte) ডাটা সংরক্ষন করে। অন্যদিকে ক্যারেকটার টাইপ অ্যারে কে স্ট্রিং(String) বলা হয়। অর্থাৎ যদি একাধিক ক্যারেকটার একত্রে কোন নির্দিষ্ট নিয়মে সাজানো হয় তাহলে এই ক্যারেকটারগুলোকে একত্রে একটি স্ট্রিং বলা হয়। যেমন ঃ “Steve Jobs”, “Swift”, “Programming”, “Macbook Pro and My IPhone” ইত্যাদি। অনেকসময় একটি ক্যারেকটার কে ও স্ট্রিং হিসেবে ব্যবহার করা যায় যার উদাহরন আমরা একটু পরেই দেখব।

অন্যান্য প্রোগ্রামিং ল্যাঙ্গুয়েজের চেয়ে সুইফ্ট এ স্ট্রিং ও ক্যারেকটার ম্যানিপুলেট করা অনেক বেশী ফাস্টার এবং সুইফ্ট ক্যারেকটার গুলোতে ইউনিকোড ক্যারেকটার স্টোর করে। অর্থাৎ যেকোন ক্যারেকটারে ভ্যালু হিসেবে ইউনিকোড স্টোর করা যায় যা এনকোডিং ডিকোডিং এর ঝামেলা কমিয়ে দিয়েছে। এছাড়া সুইফ্ট ল্যাঙ্গুয়েজে স্ট্রিং ম্যানিপুলেশন খুবই সহজ। খুব সহজেই যেকোন স্ট্রিং এর কোন ইনডেক্সে কনস্ট্যান্ট, ক্যারেকটার, নাম্বার, এক্সপ্রেশন ইত্যাদি ইনসার্ট করা যায়। এভাবে স্ট্রিং এর মধ্যে কোন কিছু ইনসার্ট করাকে ইন্টারপুলেশন বলা হয়।

স্ট্রিং ডিক্লেয়ার করাঃ
কিভাবে ভ্যারিয়েবল ও কনস্ট্যান্ট ডিক্লেয়ার করতে হয় তা আমরা প্রথম অধ্যায়ে শিখেছি। একই ভাবে স্ট্রিং টাইপ ভ্যারিয়েবল বা কনস্ট্যান্ট ডিক্লেয়ার করতে হয়।

let stringVariable = "যেকোন ইউনিকোড সাপোর্টেড ক্যারেকটারসমুহ"
let anotherStringVariable: String = "Yosemite is the name of latest OS X"

যেহেতু stringVariable ও anotherStringVariable দুটিতে স্ট্রিং দিয়ে ইনিশিয়ালাইজ করা হয়েছে তাই এই কনস্ট্যান্ট দুটি এখন স্ট্রিং টাইপ কনস্ট্যান্ট হিসেবে কাজ করবে। এবং এদের উপর যাবতীয় স্ট্রিং অপারেশনগুলো করা যাবে।

স্ট্রিং এ ক্যারেকটার ছাড়াও অন্যান্য যা যা থাকতে পারে ঃ

  • সব ধরনের স্পেশাল (escaped special) ক্যারেকটার যেমন ঃ \০ ( null ), \n (new line), \\ (backslash), \t (horizontal tab), \” (double quote), \’ (single quote) ইত্যাদি।
  • এক বাইটের ইউনিকোড স্কেলার (\xnn, nn এর জায়গায় যেকোন দুটি হেক্সাডেসিমেল ডিজিট বসতে পারে)
  • দুই বাইটের ইউনিকোড স্কেলার (\xnnnn, nnnn এর জায়গায় যেকোন চারটি হেক্সাডেসিমেল ডিজিট বসতে পারে)
  • চার বাইটের ইউনিকোড স্কেলার (\xnnnnnnnn, nnnnnnnn এর জায়গায় যেকোন আটটি হেক্সাডেসিমেল ডিজিট বসতে পারে)

এই চারধরনের বিশেষ ক্যারেকটার গুলো নিয়ে নিচে চারটি উদাহরন দেওয়া হলঃ

let wiseWords = "\"Imagination is more important than knowledge\" - Einstein"
// "Imagination is more important than knowledge" - Einstein
let dollarSign = "\x24"        // $,  Two Byte Unicode scalar U+0024
let blackHeart = "\u2665"      // ♥,  Four Byte Unicode scalar U+2665
let sparklingHeart = "\U0001F496"  // 💖, Eight Byte Unicode scalar U+1F496

ফাঁকা স্ট্রিং ইনিশিয়ালাইজ করাঃ
সাধারনত প্রোগ্রামের শুরুতে ফাঁকা স্ট্রিং ইনিশিয়ালাইজ করা হয় যা পরবর্তীতে বিভিন্ন সময়ে বিভিন্ন অপারেশনের মাধ্যমে বড় সাইজের ক্যারেকটারের কালেকশন হয়ে কোন অর্থ বা তথ্য বহন করে। দুইভাবে এরকম ফাঁকা স্ট্রিং ইনিশিয়ালাইজ করা যায়। ভ্যারিয়েবল বা কনস্ট্যান্টে সরাসরি কোন ফাঁকা স্ট্রিং অ্যাসাইন করে অথবা String টাইপের নতুন ইনস্ট্যান্স ইনিশিয়ালাইজ করে ফাঁকা স্ট্রিং ইনিশিয়ালাইজ করা হয়। নিচে দুই ধরনের পদ্ধতির উদাহরন দেওয়া হলঃ

var ফাঁকাস্ট্রিং = ""               // empty string literal
var emptyString = ""
var anotherEmptyString = String()  // initialiser syntax
// these three strings are all empty, and are equivalent to each other

প্রয়োজনে খুব সহজেই isEmpty প্রোপার্টির বুলিয়ান ভ্যালু দেখে কোন স্ট্রিং ফাঁকা কিনা তা চেক করা যায়।

var emptyString = ""
if emptyString.isEmpty {
    println("Nothing to see here")
}
// prints "Nothing to see here"

স্ট্রিং মিউট্যাবিলিটি (String Mutability) ঃ
কোন স্ট্রিং মিউট্যাবল( Mutable or Modifiable) হবে কিনা তা নির্ধারিত হয় ভ্যারিয়েবল বা কনস্ট্যান্ট ডিক্লেয়ার করার সময়। যদি স্ট্রিংটি “var” টাইপ দিয়ে ডিক্লেয়ার করা হয় তাহলে এটিকে মডিফাই করা যাবে। কিন্তু যদি এই স্ট্রিংটি “let” টাইপ হয় তাহলে এটি একটি কনস্ট্যান্ট এর ন্যায় আচরন করবে। তাই “let” টাইপ স্ট্রিং মিউট্যাবল না। নিচের উদাহরন টি দেখলেই পরিস্কার হয়ে যাবে ব্যাপার টি।

var variableString = "Swift is"
variableString += " a new programming language."
// variableString is now "Swift is a new programming language."
 
let constantString = "Swift is very much"
constantString += " faster and interactive"
// this reports a compile-time error - a constant string cannot be modified

নোটঃ
সুইফ্ট (Swift) এর এই স্ট্রিং এর সাথে অবজেকটিভ-সি (Objective-C) এর NSString রয়েছে ব্রিজ কানেকশন। আমরা যদি Cocoa অথবা Cocoa Touch এর Foundation ক্লাস নিয়ে কাজ করি তাহলে সুইফ্ট এর স্ট্রিং টাইপ ভ্যারিয়েবলগুলোর জন্য NSString ক্লাসের সব এপিআই (API) কল করা যায়।

Cocoa বা Cocoa Touch এর Foundation ক্লাসের NSString এর ইনস্ট্যান্স তৈরী করে যদি ফাংশন বা মেথডে পাঠানো হয় তাহলে মুলত ওই স্ট্রিং এর রেফারেন্স বা পয়েন্টারকেই পাঠানো হয় যা নতুন পয়েন্টার বা রেফারেন্সে অ্যাসাইন করা হয়। আসল স্ট্রিং এর কোন কপি তৈরী হয় না। অন্যদিকে Swift এর String ভ্যালু তৈরী করে যদি কোন ভ্যারিয়েবলে অ্যাসাইন করা হয় অথবা কোন মেথড বা ফাংশনে পাঠানো তাহলে প্রথমে ওই স্ট্রিং এর একটি কপি তৈরী হয় এবং তারপর এই নতুন স্ট্রিং টি মেথডে পাঠানো হয় অথবা নতুন ভ্যারিয়েবলে অ্যাসাইন করা হয়।
আমাদের কাগুজে বইতে এই সম্পর্কে আরও বিস্তারিত থাকবে।

ক্যারেকটার ম্যানিপুলেশনঃ
এতক্ষনে আমরা জেনে গেছি যে, স্ট্রিং আসলে একত্রে থাকা অনেকগুলো Character যা সুইফ্ট এ String টাইপ দিয়ে রিপ্রেজেন্ট করা হয়। প্রত্যেকটি Character আবার একটি ইউনিকোড ক্যারেকটার। সুইফ্টে for-in লুপ দিয়ে একটি স্ট্রিং এর প্রত্যেকটি ক্যারেকটার এক্সেস করা যায়। for-in লুপ এর সিনট্যাক্স ও বিস্তারিত পরের অধ্যায়ে থাকবে। এই মুহুর্তে আমরা একটি স্ট্রিং এর প্রথম থেকে শেষ পর্যন্ত সবগুলা Character এক্সেস করব এবং তা দেখব।

for ch in "Macintosh" {
    println(ch)
}
// M
// a
// c
// i
// n
// t
// o 
// s
// h

আবার কোন এক ক্যারেকটারের স্ট্রিং থেকে Character টাইপ ভ্যারিয়েবল বা কনস্ট্যান্ট ডিক্লেয়ার করার জন্য নিচের মত কোড লিখতে হয়।

let charConstant: Character = "$"
var charVariable: Character = "ক"

স্ট্রিং এ থাকা মোট ক্যারেকটারের সংখ্যা জানাঃ
countElements() একটি গ্লোবাল মেথড যা আর্গুমেন্ট হিসেবে একটি স্ট্রিং নেয় এবং এই স্ট্রিং এ কয়টি ক্যারেকটার আছে তা রিটার্ন করে।

let intro: String = "In Swift, It is too easy to count characters."
var countChar = countElements(intro); 
println(countChar)  // 45

নোটঃ
সুইফ্ট এর স্ট্রিং এর ক্যারেকটারগুলো ইউনিকোড ক্যারেকটার হয় এবং ইউনিকোডে বিভিন্ন ক্যারেকটারের সাইজ ও আলাদা হয়। একারনে সুইফ্ট ল্যাঙ্গুয়েজে সকল ক্যারেকটার একই সাইজের মেমরী নেয় না। তাই countElements() মেথডের রিটার্ন ভ্যালু ও আমাদের চাওয়া অনুযায়ী হয় না। তাই কোন স্ট্রিং এ ক্যারেকটারের সঠিক সংখ্যা জানার জন্য লুপ ব্যবহার করে প্রত্যেকটি ক্যারেকটার গননা করতে হয়। নিচের উদাহরনটিতে প্রথম লাইনে counter ভ্যারিয়েবল ০ দিয়ে ইনিশিয়ালাইজ করা হয়েছে। এরপর for-in লুপ দিয়ে প্রথম থেকে শেষ পর্যন্ত প্রত্যেকটি ক্যারেকটার ভিজিট করা হয় এবং প্রতিবার counter এর ভ্যালু ১ করে বাড়ানো হয়। ফলে for-in টির এক্সিকিউশন শেষ হলে counter ভ্যারিয়েবলে “Macintosh” এর মোট ক্যারেকটারের সংখ্যা পাওয়া যাবে।

var counter = 0
for ch in "Macintosh" {
    counter++;
}
println("This string has \(counter) characters"); // This string has 9 characters

স্ট্রিং এবং ক্যারেকটার কনক্যাট (Concate) করা ঃ
স্ট্রিং ও ক্যারেকটার টাইপ ভ্যারিয়েবল বা কনস্ট্যান্ট একত্রে যুক্ত (Concatenation) করে নতুন স্ট্রিং ইনিশিয়ালাইজ করা যায়। আবার কোন স্ট্রিং এর সাথে ক্যারেকটার যুক্ত (Concate) করে স্ট্রিং টিকে মডিফাই (Modify) করা যায়। নিচের উদাহরন গুলো দেখলেই ব্যপারগুলো বোঝা যাবে।

//Example 1
let string1 = "hello"
let string2 = " Steve"
let character1: Character = "!"
let character2: Character = "?"
 
let stringPlusCharacter = string1 + character1        // equals "hello!"
let stringPlusString = string1 + string2              // equals "hello Steve"
let characterPlusString = character1 + string1        // equals "!hello"
let characterPlusCharacter = character1 + character2  // equals "!?"

//Example 2
var instruction = "follow"
instruction += string2
// instruction now equals "follow Steve"
 
var welcome = "good morning"
welcome += character1
// welcome now equals "good morning!"

স্ট্রিং ইন্টারপুলেশন (String Interpolation)ঃ
কোন স্ট্রিং এর মধ্যে নতুন করে কোন কনস্ট্যান্ট, ভ্যারিয়েবল, এক্সপ্রেশন ইত্যাদির মিশিয়ে নতুন কোন স্ট্রিং তৈরী করা কে বলা হয় স্ট্রিং ইন্টারপুলেশন। এজন্য যে নতুন ভ্যারিয়েবল বা কনস্ট্যান্ট বা এক্সপ্রেশন মেশানো হবে তা প্যারেনথেসিস বা “()” এর ভিতর লিখে তার আগে একটি ব্যাকস্ল্যাশ দিতে হয়। তাহলে স্ট্রিং এই অংশে নতুন জিনিস টি ইনসার্ট হয়ে যায়।

let multiplier = 3
let message = "\(multiplier) times 2.5 is \(Double(multiplier) * 2.5)"
// message is "3 times 2.5 is 7.5"

উপরের কোডটিতে multiplier কনস্ট্যান্টটি ২ বার ব্যবহার করে একটি নতুন স্ট্রিং message তৈরী করা হয়েছে। এজন্য যেসব পজিশনে multiplier এর ভ্যালু ইনসার্ট করা হবে সেসব স্থানে প্রথমে ব্যাকস্ল্যাশ ও তারপর প্যারেনথেসিস দিয়ে multiplier লেখা হয়েছে। ইন্টারপুলেশনের সময় “\(multiplier)” এর পরিবর্তে এখানে multiplier এর ভ্যালু তথা 3 বসবে। এখানে লক্ষ্য করার বিষয় হল প্রথম multiplier টি একটি কনস্ট্যান্ট যা সরাসরি স্ট্রিং টিতে ইনসার্ট করা হয়েছে। কিন্তু ২য় টিতে একটি এক্সপ্রেশন “Double(multiplier) * 2.5” ইনসার্ট করা হয়েছে। প্রথমে multiplier এর ভ্যালু Double দিয়ে কাস্ট (Cast) করে তারপর 2.5 দিয়ে গুন করার পর যে রেসাল্ট পাওয়া যাবে তাই ইনসার্ট করা হয়েছে স্ট্রিং টিতে।

স্ট্রিং কমপ্যারিজন (String Comparison)ঃ
সুইফ্ট (Swift) এর String টাইপের মধ্যকার কমপ্যারিজন করার জন্য ৩টি মেথড রয়েছে : স্ট্রিং(String) ইকুয়্যালিটি, প্রিফিক্স(Prefix) ইকুয়্যালিটি এবং সাফিক্স(Suffix) ইকুয়্যালিটি।

২ টি স্ট্রিং ইকুয়্যাল হবে যদি এবং কেবল যদি উভয় স্ট্রি এর ক্যারেকটারগুলো একই হয় এবং একই অর্ডারে থাকে। এক্ষেত্রে মনে রাখতে হবে বড় হাতের অক্ষর আর ছোট হাতের অক্ষরের ইউনিকোড আলাদা হওয়ায় ক্যারেকটার হিসেবেও এরা আলাদা হবে।

let quotation = "We're a lot alike, you and I."
let sameQuotation = "We're a lot alike, you and I."
if quotation == sameQuotation {
    println("These two strings are considered equal")
}
// prints "These two strings are considered equal"

স্ট্রিং টিতে কোন পার্টিকুলার (particular) স্ট্রিং প্রিফিক্স বা সাফিক্স আছে কিনা তা চেক করার জন্য রয়েছে hasPrefix বা hasSuffix মেথড। উভয় মেথড স্ট্রিং টাইপ আর্গুমেন্ট নিয়ে বুলিয়ান ভ্যালু রিটার্ন করে। আর্গুমেন্টের প্রত্যেকটি ক্যারেকটার মুল স্ট্রিং এ একই অর্ডারে আছে কিনা সেটাই চেক করে মেথড দুটি।

let comment = "Swift is a awesome language."
if comment.hasPrefix("Swift"){
   // String comment contains a "Swift" as prefix  
}

if comment.hasSuffix("language."){
   // String comment contains a "language." as Suffix  
}

উপরের উদাহরনটিতে comment একটি স্ট্রিং টাইপ কনস্ট্যান্ট নেওয়া হয়েছে। hasPrefix(“Swift”) মেথড টি মুলত comment স্ট্রিং এর প্রথমে “Swift” স্ট্রিং টি আছে কিনা। যেহেতু comment স্ট্রিং টিতে প্রিফিক্স হিসেবে “Swift” আছে তাই মেথডটি TRUE রিটার্ন করে। আবার hasSuffix(“language.”) মেথডটি এটাই চেক করছে যে comment স্ট্রিং টি “language.” দিয়ে শেষ হয়েছে কিনা। যেহেতু স্ট্রিং টির শেষ “language.” দিয়ে হয়েছে তাই মেথড টি TRUE রিটার্ন করে।

কেস(Case) কনভারসনঃ
স্ট্রিং ম্যানিপুলেশনের একটি কমন টাস্ক হল কেস কনভারসন। অর্থাৎ ছোট হাতের অক্ষর থেকে বড় হাতের অক্ষর বা বড় হাতের অক্ষর থেকে ছোট হাতের অক্ষরে রুপান্তর করা। সুইফ্টে এই কাজ যথেষ্ট সহজেই করা যায়।

let normal = "Would you mind giving me a glass of Water?"
let uppercase = normal.uppercaseString
// uppercase is equal to "WOULD YOU MIND GIVING ME A GLASS OF WATER?"
let lowercase = normal.lowercaseString
// lowercase is equal to "would you mind giving me a glass of water?"

পরিসমাপ্তিঃ
নিশ্চয়ই খেয়াল করেছেন যে আমরা বার বার ইউনিকোড শব্দটি উচ্চারন করছি। স্ট্রিং ও ক্যারেকটার হ্যান্ডলিং এর জন্য সুইফ্টের রয়েছে কিছু বিশেষ মেথড যা দিয়ে সহজেই স্ট্রিং ও ক্যারেকটারের ইউনিকোড রিপ্রেজেন্টেশন ম্যানিপুলেশন করা যায়। এসম্পর্কে বিস্তারিত আমাদের কাগুজে বইতে পাওয়া যাবে।

বইয়ের আপডেট পেতে চোখ রাখুন আমাদের ফ্যান পেজে

পরের চাপ্টারঃ 
পরের চাপ্টারে Collections তথা Arrays ও Dictionaries নিয়ে বিস্তারিত থাকবে। এছাড়াও থাকবে Enumerations ও Closures নিয়ে অল্প বিস্তর আলোচনা।

পরের চাপ্টারঃ  কালেকশনস (অ্যারে ও ডিকশনারী), ইনুমারেশন ও ক্লোজার

২-২ঃ iOS অ্যাপে ব্যাসিক ইনপুট আউটপুট ও কিবোর্ড হ্যান্ডেলিং

Standard

এই সিরিজের পুরো পোস্ট লিস্ট এবং সম্ভাব্য কাগুজে বই এর সূচি দেখতে ক্লিক করুনbangla-ios-objective-c-swift-nuhil

আগের চ্যাপ্টারঃ  ২-১ঃ টুলস সেটআপ এবং একটি সাধারণ হ্যালো ওয়ার্ল্ড অ্যাপ তৈরি

ভূমিকাঃ
এই চ্যাপ্টারে আমরা iOS অ্যাপে কিভাবে সাধারণ ডাটা ইনপুট দেয়া যায় এবং সেটা কিভাবে হ্যান্ডেল করতে হয় তা আলোচনা করবো। ডাটা ইনপুট এর সাথে সাথে যেহেতু কিবোর্ড এর ব্যাপারটাও চলে আসে তাই কিবোর্ড হ্যান্ডেলিং নিয়েও সহজ কিন্তু সবসময় কাজে লাগে এমন কিছু ফাংশনের ব্যবহার শিখবো। আর এর মাঝখানে দেখে নেব কিভাবে Apple ডকুমেন্টেশন দেখে দেখে হঠাৎ যেকোনো অচেনা এলিমেন্ট বা অবজেক্ট নিয়ে কাজ করা যেতে পারে।
নিচের উদাহরণ হিসেবে আমরা সেরকম একটি অ্যাপ ধাপে ধাপে করবো যেখানে এই বিষয় গুলোর বাস্তব প্রয়োগ দেখা যাবে। চলুন শুরু করি।

প্রোজেক্ট ও ইউজার ইন্টারফেস তৈরিঃ
প্রথমে Xcode ওপেন করে একটি নতুন প্রোজেক্ট তৈরি করুন File -> New -> Project… এ ক্লিক করে। এক্ষেত্রে টাইপ হিসেবে সিলেক্ট করুন iOS -> Application -> Single View Application. নিচের মত করে,
2-2-1
তারপরের স্ক্রিনে এভাবে,
2-2-2

প্রোজেক্ট তৈরি হবার পর Xcode এর বাম পাশ থেকে Main.storyboard ফাইলটি সিলেক্ট করুন। ডান পাশে একটিই মাত্র ভিউ ফাইল দেখা যাবে যেহেতু আমাদের অ্যাপ এর টাইপ সিঙ্গেল ভিউ। এরপর Xcode এর ডান পাশের নিচের ইন্টারফেস বিল্ডার প্যানেল থেকে মেইন ভিউ ফাইল বা স্ক্রিনের উপর একটি Label টাইপ এলিমেন্ট নিন,
Screen Shot 2014-06-19 at 2.11.20 AM
এবং সেটা সিলেক্ট থাকা অবস্থায় Xcode এর ডান পাশের Attribute Inspector ব্যবহার করে সেটার বিভিন্ন ভিজুয়াল অ্যাপেয়ারেন্স পরিবর্তন করে নিন ইচ্ছা মত।
Screen Shot 2014-06-19 at 2.12.25 AM
এরপর আমাদের একমাত্র ভিউ এর উপর একটি Text Field টাইপ এলিমেন্ট নিন এবং সেটারও বিভিন্ন প্রোপার্টি যেমন সাইজ, ফন্ট ইত্যাদি পরিবর্তন করে নিন ডান পাশের Attribute Inspector ব্যবহার করে। এভাবে ঠিক নিচের ছবির মত একটি লে-আউট তৈরি করে নিন। ধরে নিচ্ছি উপরে বলা কথা গুলো বুঝতে পেরেছেন। না বুঝলে এই সেকশনের ১ম চ্যাপ্টার ঘুরে আসুন
2-2-3

আউটলেট তৈরিঃ
আমাদের ইউজার ইন্টারফেস ডিজাইন হয়ে গেছে দুটি ইউআই UI এলিমেন্ট দিয়ে। এখন এগুলোর জন্য আউটলেট তৈরি করতে হবে যাতে করে আমাদের View Controller কোডের মধ্যে থেকে এগুলোর রেফারেন্স পাওয়া যায়। এজন্য প্রথমে Xcode এর ডান দিকের উপর পাশ থেকে Assistant Editor বাটনটি এনাবেল করে নিন (নিচের ছবিতে 1 চিহ্নিত)। এনাবেল হলে ভিউ ফাইলের পাশেই আরও একটি এরিয়া তৈরি হবে যেখানে ViewController.h ফাইলটি ওপেন অবস্থায় থাকার কথা। ওই ফাইল ওপেন না থাকলে নিচের ছবিতে 2 চিহ্নিত জায়গাটায় ক্লিক করে ViewController.h ফাইলকে ওখানে ওপেন করতে পারেন। এখন কিবোর্ডের Control কি চেপে ধরে আমাদের ভিউ ফাইলের টেক্সট ফিল্ডের উপর থেকে মাউস ক্লিক চেপে ধরে ডান পাশের ফাইলের @interface এর নিচে যেকোনো জায়গায় ছেড়ে দিন। নিচের মত একটি ছোট পপ-আপ আসবে যেখানে এটার প্রোপার্টি টাইপ, নাম ইত্যাদি ঠিক করে দিতে পারবেন।
2-2-5

এভাবে সুইচ এলিমেন্ট, ম্যাসেজ দেখানোর লেবেল এবং বাটনটির প্রোপার্টিও ঠিক করে দিন নিচের কোডের মত,

// ViewController.h

#import <UIKit/UIKit.h>

@interface ViewController : UIViewController
@property (strong, nonatomic) IBOutlet UITextField *courseTitle;
@property (strong, nonatomic) IBOutlet UISwitch *continueCourse;
@property (strong, nonatomic) IBOutlet UILabel *messageBox;
@property (strong, nonatomic) IBOutlet UIButton *showButton;

@end

কিবোর্ড হ্যান্ডেলিংঃ
এখন পর্যন্ত ইন্টারফেস এলিমেন্ট গুলো এবং তাদের আউটলেট গুলো রেডি। এই অবস্থায় অ্যাপটি রান করে দেখতে পারেন। যদি সব ঠিক ঠাক ঠাকে তাহলে টেক্সট ফিল্ডটিতে কিছু লেখার জন্য ট্যাপ/ক্লিক করলে সিমুলেটরের কিবোর্ডটী চলে আসবে এবং সেটা ব্যবহার করে কিছু লিখতে পারবেন। কিন্তু লেখা শেষে দেখবেন কিবোর্ডটী সিমুলেটর থেকেই যাচ্ছে। আড়ালে চলে যাচ্ছে না।
এখন আমরা কিবোর্ড হ্যান্ডেলিং এর এই সমস্যার একটা সমাধান করবো। এ জন্য প্রথমে ViewController.h ফাইলে textFieldReturn: নামের একটি ফাংশন ডিক্লেয়ার করবো। মূলত এই ফাংশনের মাধ্যমে আমরা কিবোর্ড এর রিটার্ন কি চেপে কিবোর্ডকে হাইড করে ফেলার একটা উপায় প্রয়োগ করবো।

// ViewController.h

#import <UIKit/UIKit.h>

@interface ViewController : UIViewController
@property (strong, nonatomic) IBOutlet UITextField *courseTitle;
@property (strong, nonatomic) IBOutlet UISwitch *continueCourse;
@property (strong, nonatomic) IBOutlet UILabel *messageBox;
@property (strong, nonatomic) IBOutlet UIButton *showButton;

-(IBAction)textFieldReturn:(id)sender;

@end

এখন এই ফাংশনটির ইমপ্লিমেন্টেশন লিখবো ViewController.m ফাইলে নিচের মত করে,

// ViewController.m

#import "ViewController.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
	// Do any additional setup after loading the view, typically from a nib.
}

-(IBAction)textFieldReturn:(id)sender
{
    [sender resignFirstResponder];
}

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

উপরের textFieldReturn: মেথডের মধ্যে থেকে আমরা যা করছি তা হল, ইভেন্টটি যে অবজেক্ট দারা ট্রিগারড হয়েছে সেই অবজেক্টের resignFirstResponder মেথডকে কল করছি। First Responder হচ্ছে সেই অবজেক্ট যেটা এই মুহূর্তে ইউজারের সাথে ইন্টারঅ্যাক্ট করছে, এ ক্ষেত্রে কিবোর্ডটাই ফার্স্ট রেস্পন্ডার।

এখন textFieldReturn: মেথডটি যাতে সঠিক সময় কল হয় সেজন্য কিছু কাজ করতে হবে। ভিউ কন্ট্রোলার ফাইলের টেক্সট ফিল্ডটিকে সিলেক্ট করুন এবং Xcode এর ডান পাশ থেকে Connection Inspector সিলেক্ট করুন। এরপর Did End on Exit নামের সার্কেল থেকে মাউস ক্লিক করে টেনে এনে ভিউ ফাইলের নিচের View Controller আইকনের উপর ছেড়ে দিন এবং সেখান থেকে textFieldReturn সিলেক্ট করুন। নিচের ছবির মত করে,
Screen Shot 2014-06-19 at 2.20.22 AM

এ অবস্থায় সব সেভ করে যদি অ্যাপটি রান করেন এবং টেক্সট ফিল্ডে কিছু লিখে কিবোর্ডের Return বাটনে ক্লিক/ট্যাপ করেন তাহলে কিবোর্ডটি হাইড হয়ে যাবে, যা আমরা করতে চাচ্ছিলাম।
আরও একটা সিচুয়েশনে কিবোর্ড হাইড করা ভালো ইউজার এক্সপেরিএন্স যেমন, টেক্সট ফিল্ড বাদে ভিউ এর অন্য কোথাও ট্যাপ করলেও যাতে কিবোর্ডটি হাইড হয়ে যায়। চলুন সেই ব্যবস্থা করি।
এর জন্য আমরা touchesBegan: নামক ইভেন্ট হ্যান্ডেলার মেথডটি ইমপ্লিমেন্ট করবো অর্থাৎ স্ক্রিনে টাচ হলেই এটি সক্রিয় হবে। কিন্তু এই মেথডের মধ্যে আবার এটিও চেক করতে হবে যাতে কেবল মাত্র আমাদের টেক্সট ফিল্ড বাদে অন্য কোথাও টাচ হলেই কিবোর্ড হাইডের মেথড কল করতে পারি। নিচের মত করে মেথডটি ViewController.m ফাইলে লিখে ফেলুন। তাহলে আপডেটেড ফাইলটি হল,

// ViewController.m

#import "ViewController.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
	// Do any additional setup after loading the view, typically from a nib.
}

-(IBAction)textFieldReturn:(id)sender
{
    [sender resignFirstResponder];
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
    
    UITouch *touch = [[event allTouches] anyObject];
    if ([self.courseTitle isFirstResponder] && [touch view] != self.courseTitle) {
        [self.courseTitle resignFirstResponder];
    }
    [super touchesBegan:touches withEvent:event];
}

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

এখন আবার অ্যাপটি রান করুন এবং টেক্সট ফিল্ডে ক্লিক করুন। স্বভাবতই কিবোর্ড চলে আসবে। এখন হয় কিবোর্ডের return বাটন চাপুন নয়ত স্ক্রিনের যেকোনো জায়গায় ক্লিক/ট্যাপ করুন, কিবোর্ড হাইড হয়ে যাবে।

ইনপুট আউটপুটঃ
ওকে, এবার আসুন ডাটা ইনপুট এবং সেটা স্ক্রিনে আউটপুটের ব্যবস্থা করা যাক। ViewController.h ফাইলে নতুন একটি ফাংশন ডিক্লেয়ার করুন যাতে আপডেটেড ফাইলটি দেখতে নিচের মত হয়,

// ViewController.h

#import <UIKit/UIKit.h>

@interface ViewController : UIViewController
@property (strong, nonatomic) IBOutlet UITextField *courseTitle;
@property (strong, nonatomic) IBOutlet UISwitch *continueCourse;
@property (strong, nonatomic) IBOutlet UILabel *messageBox;
@property (strong, nonatomic) IBOutlet UIButton *showButton;

-(IBAction)textFieldReturn:(id)sender;
-(IBAction)outputData;

@end

এবার এই outputData মেথডটির ইমপ্লিমেন্টেশন লিখে ফেলুন ViewController.m ফাইলে যাতে পুরো ফাইলটি দেখতে নিচের মত হয়,

// ViewController.m

#import "ViewController.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
	// Do any additional setup after loading the view, typically from a nib.
}

-(IBAction)textFieldReturn:(id)sender
{
    [sender resignFirstResponder];
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
    
    UITouch *touch = [[event allTouches] anyObject];
    if ([self.courseTitle isFirstResponder] && [touch view] != self.courseTitle) {
        [self.courseTitle resignFirstResponder];
    }
    [super touchesBegan:touches withEvent:event];
}

- (IBAction)outputData {
    NSString *isCourseOngoing = (self.continueCourse.on)? @"Ongoing" : @"Not Ongoing";
    
    self.messageBox.text = [NSString stringWithFormat:@"%@ is %@", self.courseTitle.text, isCourseOngoing];
}

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

এক্সট্রা টিপস (Apple ডকুমেন্টেশন ফলো করা) ঃ
উপরে খেয়াল করুন, outputData মেথডের মধ্যে প্রথমে আমরা isCourseOngoing -এ একটা স্ট্রিং ভ্যালু সেট করছি যেটা নির্ভর করছে continueCourse নামের আউটলেট প্রোপার্টির অর্থাৎ সুইচ এলিমেন্টটির বর্তমান অবস্থার উপর। এখন কথা হচ্ছে এই যে এখানে continueCourse.on লিখে ওই অবজেক্টটির স্ট্যাটাস অ্যাক্সেস করলাম সেটা হঠাৎ আমরা জেনে নিতে পারি কই থেকে?

খেয়াল করলে দেখবেন ওই অবজেক্টটি একটি UISwitch টাইপের। অর্থাৎ এর বিস্তারিত জানা যাবে UISwitch ক্লাসের রেফারেন্স ঘেঁটে দেখলেই। আপনি ঠিকি ধরেছেন, এরকম যেকোনো ক্লাসের রেফারেন্স দেখে নিয়ে সেটার অবজেক্টের উপর আমরা বিভিন্ন অপারেশন করতে পারি। যেমন এই ক্লাসের রেফারেন্স থেকে আমরা জানতে পারি এর কি কি প্রোপার্টি ও মেথড আছে এবং সেরকম একটা প্রোপার্টি হচ্ছে on প্রোপার্টি। এই লিঙ্কে গেলেই দেখতে পারবেন লেখা আছে নিচের মত,
Screen Shot 2014-06-19 at 1.25.56 AM

অর্থাৎ আমরা continueCourse.on এর মাধ্যমে ওই প্রোপার্টির ভ্যালু অ্যাক্সেস করে ঠিক করতে পারি সুইচটি কি অন নাকি অফ অবস্থায় আছে।
তার ঠিক পরেই আমরা messageBox লেবেল এর text প্রোপার্টি হিসেবে একটি ফরম্যাটেড স্ট্রিং সেট করছি; courseTitle এর text প্রোপার্টি এবং isCourseOngoing এর ভ্যালু মিলিয়ে।

এবার শেষ বারের মত অ্যাপটি রান করুন আর টেক্সট ফিল্ডে যেকোনো ভ্যালু এবং তার নিচের সুচটি অন/অফ করে Show Me Back বাটনে ক্লিক করে দেখুন বাটনের ঠিক উপরের জায়গায় আপনার মন মত আউটপুট দেখাচ্ছে কিনা।

বইয়ের আপডেট পেতে চোখ রাখুন আমাদের ফ্যান পেজে

পরের চ্যাপ্টারঃ
পরের চ্যাপ্টারে অটো লে-আউট নিয়ে বিস্তারিত আলোচনা থাকবে এবং তার উপর ভিত্তি করে একটি পুরো উদাহরণ থাকবে আর তার পর পরই আসবে টেবিল ভিউ নিয়ে চ্যাপ্টার।

৩-১ঃ সুইফ্ট (Swift) – অ্যাপলের নতুন চমক (পরিচিতি ও অন্যান্য বেসিক)

Standard

এই সিরিজের পুরো পোস্ট লিস্ট এবং সম্ভাব্য কাগুজে বই এর সূচি দেখতে ক্লিক করুনbangla-ios-objective-c-swift-nuhil

ভুমিকাঃ
অবজেকটিভ-সি (Objective-C) এর বেসিক লার্নিং (সেকশন ১) ও আইওএস জিইউআই (GUI) অ্যাপ ডেভেলপমেন্ট (সেকশন ২) এ নজর রাখার জন্য আপনাকে অসংখ্য অভিনন্দন। আমাদের সেকশন ১ লেখা শেষ হওয়ার সাথে সাথেই অনুষ্ঠিত হয় অ্যাপলের ওয়ার্ল্ড ওয়াইড ডেভেলপার কনফারেন্স – ২০১৪ (WWDC – 2014)। প্রতিবছরের মত এবারও অ্যাপলের ওয়ার্ল্ডওয়াইড ডেভেলপার কনফারেন্স – ২০১৪ (WWDC) -এ ডেভেলপাররা পেয়েছে ৩ টি নতুন চমক। একদিকে আইওস-৮ (iOS 8) ও ইউসেমাইট (Yosemite, OS X 10.1০) এবং অন্যদিকে সুইফ্ট (Swift)। সুইফ্ট হল একেবারেই নতুন একটি প্রোগ্রামিং ল্যাঙ্গুয়েজ যা দিয়ে অবজেক্টিভ-সি এর মতই আইফোন, আইপ্যাড ও আইপডের জন্য অ্যাপ্লিকেশন ডেভেলপ করা যাবে। প্রায় ৩০ বছর ধরে অবজেক্টিভ-সি একাই রাজত্ব করে আসছে আইওএস অ্যাপ্লিকেশন ডেভেলপমেন্ট ফিল্ডে। অ্যাপলের মতে অদুর ভবিষ্যতে অবজেকটিভ-সি কে সরিয়ে এই রাজত্ব দখল করে নেবে সুইফ্ট নিজের ফাস্ট (first), মডার্ন (modern), সেইফ (safe), ইন্টারঅ্যাকটভ (interactive) বৈশিষ্ট্যের দ্বারা।

সুইফ্ট – প্রোগ্রামিং ল্যাঙ্গুয়েজ ঃ
আপনার যদি ওয়েব ডেভেলপমেন্ট (জাভাস্ক্রিপ্ট ও অন্যান্য স্ক্রিপ্টিং ল্যাঙ্গুয়েজ) সম্পর্কে ধারনা থাকে, তাহলে সুইফ্ট এর সিনট্যাক্স আপনার কাছে পরিচিত মনে হবে এবং আপনি অনেক সহজেই সুইফ্টের সিনট্যাক্স আত্মস্থ করতে পারবেন।

কিছু বিশেষ অবস্থার প্রিভিউ দেখার জন্য, সুইফ্টে লেখা প্রোগ্রাম কোন সিমুলেটর এ কম্পাইল ও রান করানো লাগবে না। Xcode 6 এ প্লেগ্রাউন্ড (playground) নামক নতুন ফিচার পাবেন যা দিয়ে সহজে সুইফ্ট এ লেখা প্রোগ্রামের রিয়েল টাইম আউটপুট দেখা যাবে।

মাত্রই ঘোষিত হওয়া সুইফ্ট শেখার জন্য অ্যাপল দিয়েছে পর্যাপ্ত রিসোর্স। একটি পিডিএফ ফরম্যাটের বই (অ্যাপলের দেওয়া সুইফ্ট এর ফ্রি বই) এবং আরও আছে কিছু  অনলাইন রিসোর্স

সুইফ্ট ল্যাঙ্গুয়েজে হ্যালো ওয়ার্ল্ড প্রোগ্রামঃ
XCode 6 এ একটি নতুন প্রজেক্ট ওপেন করি। যেভাবে অবজেক্টিভ-সি ও আইফোন ডেভেলপমেন্ট প্রজেক্ট ওপেন করা হয়, সুইফ্ট এর জন্য একিভাবে প্রজেক্ট ওপেন করতে হয়। শুধুমাত্র প্রথমে Playgroud টাইপ প্রজেক্ট ওপেন করা এবং ল্যাঙ্গুয়েজ লিস্ট থেকে “swift” সিলেক্ট করতে হবে।
এই সুবিধাটি XCode 6 থেকে পাওয়া যাবে।

তাহলে চলুন এবার সুইফ্ট ল্যাঙ্গুয়েজে হ্যালো ওয়ার্ল্ড প্রোগ্রামটি লিখে ফেলিঃ

println("Hello, World")

সদ্য তৈরি করা প্রজেক্টের মুল ফাইলটিতে শুধুমাত্র এটুকু লিখেই রান করলে ডানদিকের আউটপুট উইন্ডোতে “Hello, world” দেখা যাবে।

লাইব্রেরী ইমপোর্ট (Import) করা
সুইফ্ট ল্যাঙ্গুয়েজ দিয়ে করা প্রোগ্রামেও যদি কোন এক্সটার্নাল লাইব্রেরী প্রয়োজন হয় তাহলে প্রজেক্টটিতে ওই লাইব্রেরী ইমপোর্ট করতে হয়। লাইব্রেরী ইমপোর্ট করার জন্য শুধুমাত্র import দিয়ে লাইব্রেরীর নাম লিখলেই হয়।

import Foundation

println("Foundation Library is imported.");

ভ্যারিয়েবল, কনস্ট্যান্ট ও টাইপ অ্যানোটেশনঃ 
অবজেক্টিভ-সি (Objective-C) এর মত ভ্যারিয়েবল বা কনস্ট্যান্ট ডিক্লেয়ার করার জন্য সুইফ্ট এ আসল ডাটাটাইপ লেখা লাগে না। সুইফ্ট এ ভ্যারিয়েবল ডিক্লেয়ার করার জন্য শুধুমাত্র “var” ও কনস্ট্যান্ট ডিক্লেয়ার করার জন্য “let” ব্যবহার করা হয়। সুবিধা হল ভ্যারিয়েবল বা কনস্ট্যান্ট ডিক্লেয়ার করার সময় এর ডাটাটাইপ সম্পর্কে মাথা ঘামানোর প্রয়োজন নেই। যে ধরনের ডাটা দিয়ে ইনিশিয়ালাইজ করা হবে, এদের ডাটাটাইপ ও তাই হবে। নিজের প্রোগ্রামটুকু দেখলেই পরিস্কার হয়ে যাবে।

println("Hello, world")
var myVariable = 42
// myVariable is a variable which will be inferred to be int

var (x, y, z) = (11, 12, 45)                      // x = 11, y = 12, z = 45
// x, y and z variables are inferred to be int

let π = 3.1415926                                 // constant
// π is inferred to be int 

let (x, y) = (10, 20.8)                             // x = 10, y = 20.8
// x is inferred to be int and y is inferred to be double

উপরের কোড থেকে দেখা যাচ্ছে যে parentheses “( )” ব্যবহার করে একসাথে একাধিক ভ্যারিয়েবল বা একাধিক কনস্ট্যান্ট ইনিশিয়ালাইজ করা যায়। আবার কনস্ট্যান্ট এর নাম হিসেবে “π” লেখা হয়েছে। সুইফ্টে ভ্যারিয়েবল বা কনস্ট্যান্ট অথবা অন্যান্য আইডেন্টিফায়ারের নাম গুলোতে যেকোন স্পেশাল ক্যারেকটার এমনকি ইমোজি (emoji) আইকন ও ব্যবহার করা যায়। উদাহরনস্বরুপ,

let π = 3.14159       // any special characters
let 你好 = "你好世界"  // any unicode supported characters
let স্টিভ = "একজন কিংবদন্তীর নাম"  // any unicode supported characters
let 🐶🐮 = "dogcow"     // any emoji icons

emoji ক্যারেকটারের লিস্ট পাওয়ার জন্য কন্ট্রোল + কমান্ড + স্পেসবার একসাথে চাপ দিন।

আপনি চাইলে ভ্যারিয়েবল বা কনস্ট্যান্ট ডিক্লেয়ার করার সময় এক্সপ্লিসিটলি ডাটাটাইপ জানিয়ে দিতে পারেন। এজন্য ভ্যারিয়েবল বা কনস্ট্যান্ট এর নামের পর কোলন (ঃ) ও তারপর স্পেস দিয়ে ডাটাটাইপ লিখতে হয়। এভাবে আলাদা করে ডাটাটাইপ সম্পর্কে জানানো কে বলা হয় টাইপ অ্যানোটেশন (Type Annotation)।

var annotatedMessage: String = "Hello World! This variable is Type Annotated"

আচ্ছা, আমরা কি খেয়াল করেছি যে, উপরের প্রোগ্রামগুলোর কোন স্টেটমেন্টের শেষে সেমিকোলন নেই। অন্যান্য প্রোগ্রামিং ল্যাঙ্গুয়েজের মত সুইফ্টে স্টেটমেন্টের শেষে সেমিকোলন লাগে না। কিন্তু আপনার যদি ইচ্ছা হয় তাহলে দিতে পারেন। এতে কোন রকম এরর হবে না। আবার যদি আপনি একটি লাইনে একাধিক স্টেটমেন্ট লিখতে চান সেক্ষেত্রে অবশ্যই স্টেটমেন্টগুলোর শেষে সেমিকোলন দিতে হবে।

println("I am the first statement. I need semicolon"); println("I am the last statement. I don't need semicolon")
println("Another Line, I don't need any semicolon.But you can give me a semicolon if you want.");

অপারেটরঃ
অন্যান্য প্রোগ্রামিং ল্যাঙ্গুয়েজের অপারেটর গুলোর সাথে সুইফ্ট এর অপারেটর গুলোর মধ্যে তেমন কোন পার্থক্য নেই। তবে কিছু কিছু ক্ষেত্রে অনেকটা ইমপ্রুভমেন্ট এনেছে সুইফ্ট। নিচে ক্যাটেগরী অনুযায়ী কিছু অপারেটর সম্পর্কে ধারনা দেওয়া হলঃ

অ্যাসাইনমেন্ট অপারেটর (=)
অ্যাসাইনমেন্ট অপারেটর (=) একটি বাইনারি অপারেটর। অর্থাত “=” এর দুইটি অপারেন্ড (Operands) থাকবে এবং ডানদিকের ভ্যালু বামদিকের ভ্যারিয়েবল বা কনস্ট্যান্ট এ অ্যাসাইন হবে।

let a = 5
let b = 10
var c = a + b
var (x, y) = (a, b)   // a is assigned to x and b is assigned to y
var httpNotFoundError = (404, "Page Not Found")

শেষ লাইন দুটি দেখে বোঝা যাচ্ছে যে, এভাবে “=” অপারেটরের ডানদিকে যদি একাধিক ডাটা থাকে তাহলে তা বামদিকের একাধিক সংখ্যক ভ্যারিয়বল বা কনস্ট্যান্ট এ অ্যাসাইন হতে পারে। আবার কোন একটি ভ্যারিয়েবল বা কনস্ট্যান্ট এর ভ্যালু একাধিক ভ্যালুর টাপল [ var a = (b,c) ] হতে পারে।

অ্যারাথমেটিক অপারেটর
অন্যান্য সকল প্রোগ্রামিং ল্যাঙ্গুয়েজের অ্যারাথমেটিক অপারেটর গুলোর মতই সুইফ্ট এর অ্যারাথমেটিক অপারেটর গুলো নিজ নিজ অপারেশন করে।

1 + 2       // equals 3 (Addition)
5 - 3       // equals 2 (Subtraction)
2 * 3       // equals 6 (Multiplication)
10.0 / 2.5  // equals 4.0 (Division)
9 % 4       // equals 1 (Remainder after division)
-9 % 4      // equals -1 (Remainder after division)
8 % 2.5     // equals 0.5

“+” অপারেটর দিয়ে স্ট্রিং কনক্যাটেনেশনও করা যায়।

var msg1 = "Hello"
var msg2 = "World"
var concatenated = msg1 + msg2
println(concatenated)    // Hello World

অন্যান্য অ্যারাথমেটিক অপারেটর

var a = 10;
++a      // equals 11
a++      // equals 11

// currenlty a is 11
var b = a++  // b = 11, but a = 12
// at first a is assigned to b. So, value of b is same as current a (11). Then a is incremented by 1. So, value of as is 12. 

//currently a is 12
var b = ++a  // b = 13, and a = 13
// at first a is incremented by 1. So, value of a is 13 (12+1). Then value of a is assigned to b. So, value of b is also 13

//currenly a is 13
var b = a--  // b = 13, but a = 12
// at first a is assigned to b. So, value of b is same as current a (13). Then a is decremented by 1. So, value of as is 12. 

//currently a is 11
var b = --a  // b = 11, and a = 11
// at first a is decremented by 1. So, value of a is 13 (12+1). Then value of a is assigned to b. So, value of b is also 13

কমপ্যারিজন/কন্ডিশনাল অপারেটর
সুইফ্ট সকল ধরনের কন্ডিশনাল অপারেটর গুলো সাপোর্ট করে। নিচে কয়েকটি অপারেটরের নাম দেওয়া হলঃ

  • Equal to (a == b)
  • Not equal to (a != b)
  • Greater than (a > b)
  • Less than (a < b)
  • Greater than or equal to (a >= b)
  • Less than or equal to (a <= b)
1 == 1   // true, because 1 is equal to 1
2 != 1   // true, because 2 is not equal to 1
2 > 1    // true, because 2 is greater than 1
1 < 2    // true, because 1 is less than 2
1 >= 1   // true, because 1 is greater than or equal to 1
2 <= 1   // false, because 2 is not less than or equal to 1

//ternary conditional operator
let hasHeader = true
let contentHeight = 100
let rowHeight = contentHeight + (hasHeader ? 50 : 20)  // rowHeight will be 50

এধরনের অপারেটর গুলো প্রোগ্রামের বিভিন্ন সিলেকশন বা ডিসিশন নেওয়ার সময় ব্যবহৃত হয়। পরবর্তী অধ্যায়সমুহে এ ব্যাপারে বিস্তারিত আলোচনা হবে।

এছাড়াও রয়েছে লজিক্যাল অপারেটর ও রেন্জ অপারেটর। সকল অপারেটর সম্পর্কে আরও বিস্তারিত আলোচনা ও বিশ্লেষণ থাকবে আমাদের আসছে প্রিন্টেড বইয়ে।

বইয়ের আপডেট পেতে চোখ রাখুন আমাদের ফ্যান পেজে

পরের চাপ্টারঃ 
পরের চাপ্টারে সুইফ্ট ল্যাঙ্গুয়েজ দিয়ে স্ট্রিং ও ক্যারেকটার এর ম্যানিপুলেশন সম্পর্কে বিস্তারিত থাকবে। এছাড়া কিভাবে স্ট্রিং কে মিউটেবল করা হয়, কিভাবে একটি স্ট্রিং থেকে নির্দিস্ট ইনডেক্সের ক্যারেকটার খুজতে হয়, কিভাবে ইন্টারপুলেশন, কমপ্যারিজন করতে হয় ইত্যাদি সম্পর্কে থাকবে বিস্তারিত।

পরের চাপ্টারঃ  সুইফ্ট ল্যাঙ্গুয়েজে স্ট্রিং ও ক্যারেকটার টাইপ ভ্যারিয়েবল

৬-১ – অবজেক্টিভ-সি (Objective-C) নাকি সুইফ্ট (Swift) ?

Standard

এটি হচ্ছে আমাদের চলতি “বাংলায়- অবজেক্টিভ-সি, সুইফ্ট এবং iOS অ্যাপ ও গেম ডেভেলপমেন্ট” সম্পর্কিত সিরিজ পোস্ট ও প্রকাশিতব্য বইয়ের ষষ্ঠ সেকশন (কিছু সাধারণ প্রশ্ন ও উত্তর) এর প্রথম চ্যাপ্টার।

ভূমিকাঃ
আপনি যদি Apple এর WWDC (Worldwide Developer’s Conference) সম্পর্কে মোটা মুটি অবগত থাকেন অথবা ইনফরমেশন টেকনোলজি সম্পর্কিত  আন্তর্জাতিক খবর গুলো খেয়াল করে থাকেন, তাহলে জেনে থাকবেন যে Apple তাদের WWDC 2014 ইভেন্টে সবচেয়ে চমকপ্রদ যে আবিষ্কারটির ঘোষণা দিয়েছে তা হচ্ছে তাদের তৈরি সম্পূর্ণ নতুন একটি প্রোগ্রামিং ল্যাঙ্গুয়েজের খবর। যার নাম Swift. তারা চায় তাদের ভবিষ্যৎ iOS এবং OSX অ্যাপ্লিকেশন গুলো এই ল্যাঙ্গুয়েজ দিয়েই ডেভেলপ করা হোক যাতে করে এই প্ল্যাটফর্মের অ্যাপ গুলোর পারফরমেন্স আরও ভালো হয়।
এটাকে তারা বলছে, দ্রুতগতি সম্পন্ন, আধুনিক, নিরাপদ ও ইন্টার‌অ্যাক্টিভ একটি ল্যাঙ্গুয়েজ। অন্যান্য ল্যাঙ্গুয়েজের মত অনেক অনেক জনপ্রিয় ফিচার এই ল্যাঙ্গুয়েজে যুক্ত আছে। এর ডিজাইন এমন ভাবে করা হয়েছে যাতে সিনট্যাক্স আরও সহজ হয় এবং iOS ও OSX ডেভেলপমেন্ট শুরু করতে নতুনদের বাধা আরও কম হয়। এমনকি আসছে সেপ্টেম্বর, ২০১৪ তে যে Xcode 6 লঞ্চ হতে যাচ্ছে তার সঙ্গে Playground নামের একটি ফিচার থাকছে যার মাধ্যমে বিভিন্ন কোড, প্রোগ্রামিং লজিক এবং ক্যালকুলেশনের লাইভ প্রিভিউ দেখা যাবে পুরো প্রোগ্রাম রান না করেই। অর্থাৎ Apple বরাবরই ডেভেলপার ফ্রেন্ডলি একটা ডেভেলপমেন্ট প্ল্যাটফর্ম দেয়ার ব্যাপারে সবসময় গুরুত্ব দিয়েছে যারই বহিঃপ্রকাশ হিসেবে Swift এর জন্ম বলতে পারেন। অতএব, ভয় না পেয়ে এর কাছ থাকে ভালো কিছুই আশা করতে পারেন নতুন এবং পুরনো iOS এবং OSX ডেভেলপারেরা।

আমি এই প্ল্যাটফর্মে নতুন, আমার কি এখন অব্জেক্টিভ-সি অথবা সুইফ্ট নাকি দুটো ল্যাঙ্গুয়েজ-ই শেখা উচিত?
প্রথমত, সুইফ্ট (Swift) একটি নতুন প্রোগ্রামিং ল্যাঙ্গুয়েজ, আর তাই এটাতে আরও নতুন নতুন ফিচার যুক্ত হওয়া থেকে শুরু করে বিভিন্ন বাগ ফিক্সিং চলতেই থাকবে সামনের অন্তত এক দুই বছর। আর তাই Apple এটার ব্যাপারে প্রচার চালিয়ে যাবে ঠিকই কিন্তু আপনাকে বাধ্য করবে না iOS এর অ্যাপ শুধুমাত্র Swift এ করার জন্য। আর অন্যদিকে অবজেক্টিভ-সি রাতারাতি বন্ধও হয়ে যাবে না।
দ্বিতীয়ত, ইতোমধ্যে Apple অ্যাপ স্টোরে ১০ লাখেরও বেশি অ্যাপ্লিকেশন আছে যেগুলো অবজেক্টভ-সি তে করা এবং ওয়েবে কয়েক লাখ জনপ্রিয় লাইব্রেরি, ফ্রেমওয়ার্ক ওপেন সোর্স টুলস ও প্রজেক্ট আছে যেগুলোও অবজেক্টিভ-সি তে ডেভেলপ করা। আর তাই এগুলোর এনহ্যান্সমেন্ট, বাগ ফিক্সিং এবং আপগ্রেড চলবে আরও অনেক দিন আর তার জন্য অবশ্যই অব্জেক্টিভ-সি তে অভিজ্ঞ ডেভেলপার বা প্রোগ্রামারের প্রয়োজন থাকছেই।
তৃতীয়ত, Swift এবং iOS 7,8 সাথে Xcode 6 এমন ভাবে প্রস্তুত আছে যে আপনি একটি প্রোজেক্টে একি সাথে অবজেক্টিভ-সি এবং সুইফ্ট ল্যাঙ্গুয়েজ ব্যবহার করতে পারেন কোন রকম বাড়তি ঝামেলা ছাড়াই। আর এই যুগপৎ বিদ্যমানতা এটাই প্রমাণ করে যে, সুইফ্ট একবারেই অবজেক্টিভ-সি এর জায়গা দখল করে নিচ্ছে না। আরও দেখতে পারেন এখানে

আর তাই, যদি আপনি কোন iOS ডেভেলপার কোম্পানিতে জয়েন করতে চান অথবা নিজে থেকেই এই মার্কেটে অ্যাপ লঞ্চ করতে চান আপনাকে দুটো ল্যাঙ্গুয়েজেই সম্যক ধারনা নেয়া খুবি গুরুত্বপূর্ণ।

আমি বেশকিছু দিন ধরেই অবজেক্টিভ-সি তে অ্যাপ ডেভেলপমেন্ট এর কাজ করে আসছি কিন্তু এখন কি আমি একজন কেবলই নতুন শিক্ষানবিস?
একদম না। চিন্তা করে দেখুন, আপনি ইতোমধ্যে Xcode, Cocoa এবং Cocoa Touch এর বিভিন্ন API এবং অবজেক্টিভ-সি তে অভিজ্ঞতা অর্জন করেছেন যার মাধ্যমে চলছে কয়েক লাখ অ্যাপ – তার তুলনায় Swift শেখা কিছুই না। বরং আপনি আপনার অভিজ্ঞতার ভাণ্ডারে নতুন একটি জিনিষ যুক্ত করতে যাচ্ছেন মাত্র। অন্যদের থেকে তার মানে আপনি সিংহ ভাগ এগিয়ে থাকছেনই সব সময়।

সুইফ্ট দিয়ে ডেভেলপমেন্টের সুবিধা কি?
Apple এর মতে এটা ৩০ বছর বয়সী Objectiv-C এর চেয়ে অনেকটাই আধুনিক। আর তাই এতে প্রোগ্রামারদের অনেক প্রিয় কিছু ফিচার যেমন namespacing, optionals, tuples, generics, type inference ইত্যাদি থাকছে যা অবশ্যই সফটওয়্যার ডেভেলপমেন্টকে আরও বেশি যুগোপযোগী আর গুনমান সম্পন্ন করবে।
অন্যদিকে এই ল্যাঙ্গুয়েজের অবজেক্ট সর্টিং, এক্সিকিউশন সহ আরও কিছু বিষয়ে টাইম কমপ্লেক্সিটি অনেক কম।

কোথায় শেখা শুরু করবো?
সবসময় নতুন কিছু শুরু করতে বা ওই বিষয়ে জানতে সেটার অফিসিয়াল সোর্স থেকেই দেখে নেয়া উচিত। যেমন নিচের সোর্স দুটি হতে পারে সঠিক দিক নির্দেশনাঃ

আর আমরা তো আছিই। আমাদের এই সিরিজের এবং সম্ভাব্য বইয়ের দ্বিতীয় সেকশনেই থাকছে বাংলায় ব্যাসিক সুইফ্ট লার্নিং এর উপর ১০টির বেশি চ্যাপ্টার। সিরিজের সব পোষ্ট গুলোর এবং প্রিন্টেড বইয়ের আপডেট পেতে লাইক দিয়ে রাখুন আমাদের ফেসবুকে ফ্যান পেজে

আমাদের ব্লগ পোস্ট গুলোর চেয়ে অনেক বেশি বিস্তারিত আলোচনা, বিশ্লেষণ এবং কোড এক্সাম্পল থাকবে প্রিন্টেড বইয়ে।

পরের চ্যাপ্টারঃ পরের চ্যাপ্টারে থাকবে একটি সাধারণ প্রশ্ন যেটা অনেকেরই মনে জমে থাকে, “iOS এবং OSX এর অ্যাপ ডেভেলপমেন্টের জন্য Macbook, iMac, Mac mini অর্থাৎ Apple গ্যাজেট বাধ্যতামূলক কিনা” এর উপর আলোচনা এবং কিছু বিশ্লেষণ ও অবশ্যই কিছু বিকল্প ব্যবস্থার কথা।

৯- অবজেক্টিভ-সি (Objective-C) তে ব্লক (Block) এর ব্যবহার

Standard

আগের চ্যাপ্টারঃ অবজেক্টিভ-সি এর ক্যাটাগরি (CATEGORY) এবং এর বিস্তারিত

ভূমিকাঃ
সহজভাবে বলতে গেলে ব্লক হচ্ছে অবজেক্টিভ-সি এর anonymous function. Google ডেফিনেশন অনুযায়ী অ্যানোনিমাস ফাংশন হচ্ছে-
An anonymous function is a function that is not stored in a program file, but is associated with a variable whose data type is function_handle. Anonymous functions can accept inputs and return outputs, just as standard functions do. However, they can contain only a single executable statement.
এর মাধ্যমে আপনি বিভিন্ন অবজেক্টের মধ্যে ইচ্ছামত স্টেটমেন্ট (Satement) পাস তথা আদান-প্রদান করতে পারবেন যেমনভাবে সাধারণ ডাটা আদান প্রদান করে থাকেন। উপরন্তু ব্লককে ক্লোজার (Closure) হিসেবে ব্যবহার করা হয় যাতে করে তার পক্ষে তার আসে পাশের ডাটা/অবস্থা নিয়ে কাজ করাও সম্ভব হয়।
আস্তে আস্তে আমরা এ ব্যপারে বিস্তারিত জানতে চেষ্টা করব।

সিনট্যাক্সঃ
ডিক্লেয়ারেশন-

returntype (^blockName) (argumentType);

ইমপ্লিমেন্টেশন-

returntype (^blockName) (argumentType) = ^{
    // Statements
};

এখানে “=” চিহ্নের বাম পাশের অংশটি হচ্ছে একটি ব্লক ভেরিয়েবল এবং ডান পাশের অংশ হচ্ছে ব্লকটি।

উদাহরণ-

void (^simpleBlock) (void) = ^{
    NSLog(@"This is a block that does not return anything and also has no argument!");
};

উপরের এই ব্লককে আমরা simpleBlock(); এভাবে কল করতে পারি।

ব্লক (Block) তৈরিঃ
ব্লক তৈরির ব্যাপারটা মোটা মুটি ফাংশন তৈরির মতই। যেভাবে আপনি একটি ফাংশন ডিক্লেয়ার করে থাকেন ঠিক সেভাবেই একটি ব্লক ভেরিয়েবল ডিক্লেয়ার করতে পারেন আবার যেভাবে একটি ফাংশন এর ইমপ্লিমেন্ট করেন সেভাবে একটি ব্লককে ডিফাইন করতে পারেন। এমনকি যেভাবে একটি ফাংশনকে কল করেন ঠিক সেভাবেই একটি ব্লককেও কল করতে পারেন।
নিচে একটি সহজ উদাহরণ দেখানো হল,

// main.m

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[])
{

    @autoreleasepool {
        // Declare and define the isPlaceOpen block
        BOOL (^isPlaceOpen)(void) = ^ {
            return YES;
        };

        // Use the block
        NSLog(@"Open state of the place is: %hhd ", isPlaceOpen());
    }
    return 0;
}

“^” চিহ্নটি ব্যবহার করে isPlaceOpen কে একটি ব্লক হিসেবে চিহ্নিত করা হয়। এটাকে অবজেক্টিভ-সি এর “*” পয়েন্টার চিহ্নের মতই মনে করতে পারেন অর্থাৎ শুধুমাত্র ডিক্লেয়ার করার সময় এটা গুরুত্বপূর্ণ কিন্তু এর পরে এটাকে সাধারণ ভেরিয়েবলের মতই মনে করে ব্যবহার করতে পারেন।
নিচে আরেকটি উদাহরণ দেখি যেখানে আমাদের ব্লকটি দুইটি double টাইপের প্যারামিটারও গ্রহণ করে এবং সেগুলো ব্যবহার করে অল্প কিছু হিসাব করার পর একটি double টাইপ ডাটা রিটার্ন করে।

// main.m

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[])
{

    @autoreleasepool {
        // Declare the block variable
        double (^priceAfterTax)(double rawPrice, double totalTax);

        // Create and assign the block to the variable
        priceAfterTax = ^(double price, double tax) {
            return price + tax;
        };

        // Call the block
        double total = priceAfterTax(400, 40);

        NSLog(@"Total price after considering tax is: " @"%.2f Tk.", total);
    }
    return 0;
}

খেয়াল করুন, প্রথমে আমরা একটি ব্লক ভেরিয়েবল ডিক্লেয়ার করেছি তারপর আসল ব্লকটিকে সেই ভেরিয়েবলে অ্যাসাইন করেছি এবং তারপর ভেরিয়েবলের নাম ধরে সেই ব্লককে ব্যবহার বা কল করেছি। প্রোগ্রামটি রান করালে নিচের মত আউটপুট আসবে।
Screen Shot 2014-06-11 at 1.20.13 PM

ক্লোজার (Closure)ঃ
শুরুতেই বলা হয়েছে যে, সাধারণভাবে একটি ব্লক, ক্লোজার হিসেবেও ইমপ্লিমেন্টেড হয়। যেমন নিচের উদাহরণটি দেখে আমরা বিশ্লেষণ করতে পারি এখানে ক্লোজার ফিচার কোথায়,

// main.m

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[])
{

    @autoreleasepool {
        NSString *foodGenre = @"Italian";

        NSString *(^getFullItemName)(NSString *) = ^(NSString *itemName) {
            return [foodGenre stringByAppendingFormat:@" %@", itemName];
        };

        NSLog(@"%@", getFullItemName(@"Pizza"));    // Honda Accord
    }
    return 0;
}

আবারও বলি, ফাংশনের মতই একটি ব্লকের মধ্যে থেকে আপনি সেটার লোকাল ভেরিয়েবলগুলো, তার কাছে যে প্যারামিটারগুলো পাস করা হয়েছে সেগুলো এবং যেকোনো গ্লোবাল ভেরিয়েবল অ্যাক্সেস করতে পারবেন। কিন্তু ক্লোজার হিসেবে ব্লকের ব্যবহারের ফলে আপনি কিছু নন-লোকাল ভেরিয়েবলকেও ব্লকের মধ্যে থেকে অ্যাক্সেস করতে পারবেন। নন-লোকাল ভেরিয়েবল হচ্ছে ব্লকের বাইরের কোন ভেরিয়েবল কিন্তু ব্লকের Lexical Scope এর আওতাভুক্ত। উপরের উদাহরণে foodGenre সেরকম একটি নন-লোকাল ভেরিয়েবল যেটাকে getFullItemName ব্লকের মধ্যে থেকেও ব্যবহার করা গেছে।

মেথড প্যারামিটার হিসেবে ব্লক ব্যবহারঃ
একটি পূর্ণ ব্লককে যেকোনো মেথডের প্যারামিটার হিসেবেও ব্যবহার করা যায়। ওই ব্লক যে কাজটা করে মূলত সেই কাজটিকেই একটা প্যাকেজ সরূপ একটি প্যারামিটার হিসেবে যেকোনো মেথডের কাছে দেয়া যায়। নিচে সেরকম একটা উদাহরণ দেখবো আমরা।
প্রথমে আপনার প্রজেক্টকে এমনভাবে সাজিয়ে নিন যাতে সেখানে Food.h, Food.m এবং main.m এই ৩টি ফাইল থাকে নিচের মত করে,

// Food.h

#import <Foundation/Foundation.h>

@interface Food : NSObject

- (void)performActionWithCompletion: (void (^) ()) completionBlock;

- (void)calculatePriceWithTax: (double (^) (double price)) taxCalculatorBlock;

@end
// Food.m

#import "Food.h"

@implementation Food

- (void)performActionWithCompletion:(void (^) ()) completionBlock {

    NSLog(@"Started cooking...");
    completionBlock();
}

- (void)calculatePriceWithTax: (double (^) (double price)) taxCalculatorBlock {
    NSLog(@"Total price is: %f", taxCalculatorBlock(400));
}

@end
// main.m

#import <Foundation/Foundation.h>
#import "Food.h"

int main(int argc, const char * argv[])
{

    @autoreleasepool {
        Food *process = [[Food alloc]init];

        [process performActionWithCompletion: ^{
            NSLog(@"Cooking is finished and this block has been called to intimate action is performed.");
        }];

        [process calculatePriceWithTax: ^(double totalPrice) {
            return totalPrice + 40;
        }];

    }
    return 0;
}

উপরের কোড গুলো দেখুন এবং বোঝার চেষ্টা করুন, প্রথমেই আমরা Food ক্লাসের ইন্টারফেসে দুইটি মেথড ডিক্লেয়ার করেছি যে দুটো মেথড প্যারামিটার হিসেবে দুই ধরনের দুইটি ব্লক গ্রহণ করে। প্রথম মেথডটির নাম performActionWithCompletion: এবং এটি এমন একটি ব্লককে প্যারামিটার হিসেবে গ্রহণ করে যে ব্লকটির কোন কিছু রিটার্ন করে না এবং যার কোন আর্গুমেন্টও নাই। কিন্তু দ্বিতীয় মেথডটি একটি ব্লককে তার প্যারামিটার হিসেবে গ্রহণ করে যে ব্লক একটি double টাইপ ডাটা রিটার্ন করে এবং যার একটি আর্গুমেন্টও আছে।
এখন এই দুইটি মেথড এর ইমপ্লিমেন্টেশন লিখেছি Food.m ফাইলে। অর্থাৎ এই দুইটি মেথড কি কাজ করবে তার ডেফিনেশন। প্রথম মেথডটি শুরুতেই একটি ম্যাসেজ প্রিন্ট করবে এবং তারপর ওর কাছে প্যারামিটার হিসেবে আসা ব্লকটিকে কল করবে। তো, এক্ষেত্রে কি ঘটবে? ওই completionBlock নামক ব্লকের মধ্যে যা করতে বলা হয়েছে তাই ঘটবে। অর্থাৎ ওখানে লিখে রাখা আরও একটি ম্যাসেজ প্রিন্ট হবে।
দ্বিতীয় মেথড এর কাজ হচ্ছে, তার কাছে আসা ব্লকটিকে কল করবে। কিন্তু যেহেতু ওই ব্লকটির প্যারামিটার আছে তাই ওকে কল করার সময় একটি ভ্যালুও পাস করে দিবে। আর ব্লক কল করা মানে কি? ওই ব্লকের মধ্যে যা করতে বলা হয়েছে তাই করবে। এক্ষেত্রে ওই ব্লকটি একটি ডাটা রিটার্ন করে এবং আসলে calculatePriceWithTax: মেথডটি সেই রিটার্ন করা ডাটাকেই নিজের মধ্যে থেকে প্রিন্ট করে।
main.m ফাইলে আমরা উপরে উল্লেখিত মেথড দুইটি কল করেছি এবং তাদের প্যারামিটার হিসেবে দুইটি ভিন্ন রকম ব্লক কে পাঠিয়ে দিয়েছি। প্রোগ্রামট রান করালে নিচের মত আউটপুট আসবে,
Screen Shot 2014-06-11 at 2.23.24 PM

ব্লক টাইপ ডিফাইন করাঃ
typedef এর মাধ্যমে একটি ব্লক টাইপ তৈরি করা যায় যাতে করে ব্লক ব্যবহারের সময় এর সিনট্যাক্স এলোমেলো না হয়ে যায় এবং ওই ব্লকটিকে আরও সহজ এবং সংক্ষেপ নামে ব্যবহার করা যায়। এ ব্যাপারে এবং উপরের টপিক গুলোর আরও বিস্তারিত থাকবে আমাদের সম্ভাব্য কাগুজে বইয়ে। বই এর আপডেট পেতে লাইক দিয়ে রাখতে পারেন এখানে

পরের চ্যাপ্টারঃ অবজেকটিভ সি তে এক্সেপশন ও এরর হ্যান্ডেলিং