Block对象

背景:回调机制中回调设置代码和回调方法的具体实现无法写在同一段代码中。Mac OS X 10.6和iOS4种引入了Block对象。Block对象看上去是一段代码,但是可以当作数据来传递。
定义Block对象:
^{
     NSLog(@“I’m a log statement within a block!");
}
这段代码看似是一个函数,但是没有函数名,相应的位置只有一个^符号,^符号表示这段代码是一个Block对象。与函数一样,Block对象也可以有实参:
^(double dividend,double divisor){
     double quotient = dividend / divisor;
     return quotient;
}
Block对象也可以有返回值。以上Block对象都只是值,没有名称。为了能够通过名称使用某个Block对象,就必须先将其赋值给Block对象变量。
使用Block对象:
block的变量声明如下
为Block变量赋值
Block变量的值一定是一段代码
声明和赋值也可以写在一起
typedef
可以用typedef将某个block对象定义为一个新类型,以方便使用。不能在方法的实现代码中使用typedef,应该在文件的顶部或者头文件内。
如,typedef void (^ArrayEnumerationBlock)(id, NSUInterger, BOOL *);
这里定义的是一个新的类型,而不是变量。跟在^字符后面的是类型名称。定义以后可以用ArrayEnumerationBlock devowelizer;这样声明普通变量那样声明Block对象。
返回值
对于有返回值的Block对象,可以像调用函数那样调用Block对象,然后使用其返回值。
double (^divBlock)(double,double) = ^(double k, double j ) {
     return k/j;
}
这段代码首先声明了一个名为divBlock的Block变量,然后将新创建的Block对象赋给了该变量。
Block对象divBlock的示例代码如:double quotient = divBlock(42.0,12.5);
 
内存管理:
Block对象也是在栈中创建并保存的,也就是说,当创建Block对象的函数或者方法完成执行并返回后,相应的Block对象会随着栈帧地释放而被释放。当需要继续保留Block对象时,必须将Block对象从栈靠背至堆。向Block对象发送copy消息可以将其从栈拷贝至堆。
ArrayEnumerationBlock iVarDevowelizer = [devowelizer copy];
iVarDevowelizer指向的对象是基于堆的Block对象,而不是基于栈的Block对象。
在将基于栈的Block对象拷贝至基于堆的Block对象时,Block对象会捕获其使用的变量。对基本类型的变量,捕获意味着程序会拷贝变量的值,并用Blok 参对象内的局部变量保存。对指针类型的变量,Block对象会使用strong特性的引用。这意味着凡是对Block对象用到的对象,都会被保留。
strong特性的引用有可能会导致retain循环问题。当 Block 对象所使用的对象保留了当前的Block 对象时,就会发生retain循环。解决途径:现在 Block对象外声明一个_weak指针,然后将这个指针指向Block 对象使用的对象,最后在 Block 对象中使用这个新的指针。
在 Block对象中,被捕获的变量是常数,程序无法修改变量所保存的值。需要在Block对象内修改某个外部变量,可以在声明相应的外部变量时,在前面加上_block关键字。例如,以下代码可以在Block对象内将外部变量counter的值增1。
_block int counter = 0;
void (^counterBlock)( ) = ^{ counter ++ };
...
counterBlock();//counter增1,数值为2
counterBlock();//counter增1,数值为2
如果这段代码没有使用_block关键字,那么编译器会在 Block 对象的定义处报错,提示修改counter值是无效的。
 

你可能感兴趣的:(Block对象)