初学者对于block机制一时很难理解。我刚开始学习的时候也是跟着敲代码,并不能说立马理解了其中的道理。今天,我想从2个层次、5个点去相对详细的讲解一下OC中的block机制。
第一层:理解block机制
①定义(相关说明很多,了解的人可以跳过)
int a=1, b=2;
int block1 = a+b;
//定义方式1
int (^block2)(int, int) = ^(int a, int b) {
return a+b;
};
//定义方式2
int (^block3)(int, int);
block3 = ^(int a, int b) {
return a+b;
};
NSLog(@"%d %d %d", block1, block2(a, b), block3(a, b));
讲解:block可以看做一个函数。只是通常我们的函数是先声明、定义,然后再传值、调用;而block可以将定义的部分直接放在赋值表达式的一边,很方便
②OC中block如何使用?
以上的情况我们肯定不会使用block,多此一举。
首先,我们要明确,block是OC中的一种通信机制,相似的还有代理机制和KVO机制。
代理机制中:假如我们有A页面要pushViewController到B页面,把A页面的的属性通过单例或者传值方法,B页面就可以使用了。但是B页面的属性要通过Pop方式传回给A页面就要通过代理或者单例方式了。
实现一个代理要定义协议,创建代理等。有时候也许我们只是想要一个值,但通过这个方式很麻烦。如:
我在b页面有一个输入框,我在里面输入一个字符串,点击按钮跳回a页面。并希望a页面的label.text显示的内容就是b页面中输入的字符串。此时最好的方法就是用block!
第二层:深层理解block
①block中的内存管理:
block本身是像对象一样可以retain,和release。但是,block在创建的时候,它的内存是分配在栈(stack)上,而不是在堆(heap)上。他本身的作于域是属于创建时候的作用域,一旦在创建时候的作用域外面调用block将导致程序崩溃。例如:
- (void)viewDidLoad
{
[super viewDidLoad];
int number = 1;
_block = ^(){
NSLog(@number %d, number);
};
}
在一个按钮点击事件中调用:如下调用会导致程序崩溃
- (IBAction)testDidClick:(id)sender {
_block();
}
解决这个问题的方法:在创建完block的时候需要调用copy的方法。copy会把block从栈上移动到堆上,那么就可以在其他地方使用这个block了
_block = ^(){
NSLog(@number %d, number);
};
_block = [_block copy];
②循环引用:
_block = ^(){
NSLog(@string %@, _string);
};
这里的_string相当于是self->_string;那么block是会对内部的对象进行一次retain。也就是说,self会被retain一次。当self释放的时候,需要block释放后才会对self进行释放,但是block的释放又需要等self的dealloc中才会释放。如此一来变形成了循环引用,导致内存泄露。
修改方案:
①在非ARC环境中:
新建一个__block的局部变量,并把self赋值给它,而在block内部则使用这个局部变量来进行取值。因为__block标记的变量是不会被自动retain的。
__block ViewController *controller = self;
_block = ^(){
NSLog(@string %@, controller->_string);
};
②在ARC环境下:
__block要换成__weak,因为ARC环境下自动释放池会自动做引用计数的增减。此处我们需要一个弱引用对象controller指向self对象,这样即便在block中使用了controller,由于它是一个弱引用,可以使用self的地址空间但是并不会造成引用计数加1