> For the complete documentation index, see [llms.txt](https://ios.howtocode.dev/llms.txt). Markdown versions of documentation pages are available by appending `.md` to page URLs; this page is available as [Markdown](https://ios.howtocode.dev/objc-intro/9.md).

# ব্লক ও এর ব্যবহার

**ভূমিকাঃ** সহজভাবে বলতে গেলে ব্লক হচ্ছে অবজেক্টিভ-সি এর 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](http://nuhil.files.wordpress.com/2014/06/screen-shot-2014-06-11-at-1-20-13-pm.png?w=460)](https://nuhil.files.wordpress.com/2014/06/screen-shot-2014-06-11-at-1-20-13-pm.png)

**ক্লোজার (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](http://nuhil.files.wordpress.com/2014/06/screen-shot-2014-06-11-at-2-23-24-pm.png?w=460)](https://nuhil.files.wordpress.com/2014/06/screen-shot-2014-06-11-at-2-23-24-pm.png)

**ব্লক টাইপ ডিফাইন করাঃ** typedef এর মাধ্যমে একটি ব্লক টাইপ তৈরি করা যায় যাতে করে ব্লক ব্যবহারের সময় এর সিনট্যাক্স এলোমেলো না হয়ে যায় এবং ওই ব্লকটিকে আরও সহজ এবং সংক্ষেপ নামে ব্যবহার করা যায়। এ ব্যাপারে এবং উপরের টপিক গুলোর আরও বিস্তারিত থাকবে আমাদের সম্ভাব্য কাগুজে বইয়ে। বই এর আপডেট পেতে লাইক দিয়ে রাখতে পারেন [**এখানে**](https://www.facebook.com/bangla.objc.swift.ios)।

Originally Posted [Here](http://nuhil.net/2014/06/11/৯-অবজেক্টিভ-সি-objective-c-তে-ব্লক-block-এ/)


---

# Agent Instructions
This documentation is published with GitBook. GitBook is the documentation platform designed so that both humans and AI agents can read, navigate, and reason over technical content effectively. Learn more at gitbook.com.

## Querying This Documentation
If you need additional information that is not directly available in this page, you can query the documentation dynamically by asking a question.

Perform an HTTP GET request on the current page URL with the `ask` query parameter, and the optional `goal` query parameter:

```
GET https://ios.howtocode.dev/objc-intro/9.md?ask=<question>&goal=<endgoal>
```

`ask` is the immediate question: it should be specific, self-contained, and written in natural language.
`goal` is optional and describes the broader end goal you are ultimately trying to accomplish on behalf of the user. GitBook uses it to tailor the answer towards what is most useful for that goal.

The response will contain a direct answer to the question and relevant excerpts and sources from the documentation.

Use this mechanism when the answer is not explicitly present in the current page, you need clarification or additional context, or you want to retrieve related documentation sections.
