block截获self情况.
略.
block截获成员变量时,避免循环引用的几种基本操作.
下面这段代码如果不做任何处理,在block属性里直接使用成员变量,100%导致循环引用.
1 | 场景: |
那么如何解决呢?有如下几种操作:
使用__weak typeof(成员变量) weak成员变量 = 成员变量;
1
2
3
4
5_myAnimal = [[Animal alloc] init];
__weak typeof(_myAnimal) weakMyAnimal = _myAnimal;
self.blk = ^{
NSLog(@"%@", weakMyAnimal);
};使用局部变量,这样block就不再持有self.于是循环引用便不能形成了.
1
2
3
4
5_myAnimal = [[Animal alloc] init];
Animal *tMyAnimal = _myAnimal;
self.blk = ^{
NSLog(@"%@", tMyAnimal);
};对方法2进行改进,增加__block修饰符.
1
2
3
4
5
6_myAnimal = [[Animal alloc] init];
__block Animal *tMyAnimal = _myAnimal;
self.blk = ^{
NSLog(@"%@", tMyAnimal);
tMyAnimal = nil;
};使用__weak typeof(self) weakSelf = self;
1
2
3
4
5
6
7
8_myAnimal = [[Animal alloc] init];
__weak typeof(self) weakSelf = self;
self.blk = ^{
__strong typeof(weakSelf) strongSelf = weakSelf;
if (!strongSelf) return ; //必须加上判断,否则一旦self在这之前被释放销毁了,那么后面的->操作将崩溃.
NSLog(@"%@", strongSelf -> _myAnimal); //一旦block在执行中,对象被销毁了,很容易造成对NULL指针进行解引用导致崩溃.所以前面的判断是必须的.
NSLog(@"%@", (*strongSelf)._myAnimal);
};
那么这几种办法中哪种才是最佳实践呢?从写法上来看,第一种和第二种以及改进型的第三种要比第四种方便很多.那么第一种方法和第二种方法哪种更好呢?这里就需要知道什么是block以及为什么使用weak修饰符后就可以解除循环引用.
什么是block
带有自动变量(局部变量)值的匿名函数就是block.匿名函数就是不带有名称的函数.带有自动变量值在block中的表现就是”截获自动变量值”.
截获自动变量值
“截获自动变量值”指的是:在执行block语法时,block语法表达式所使用的自动变量值会被保存到block的结构体实例中(即block自身里).也就是说block结构体会多出来一个成员变量,它的类型就是截获的自动变量的类型.比如自动变量的类型是weak,那么该成员变量的类型也是weak;如果是strong,那么成员变量的类型也是strong.这就是为什么我们使用__weak typeof(self) weakSelf = self;
可以让block不强引用self的原因.
现在让我们来比较一下方法一和方法二,方法一使用了weak修饰符,因此block截获的指针变量weakMyAnimal
是weak类型的,而weak类型指针是不会强引用指向的对象,这样block也就不会强引用Animal对象.而方法二中将导致block会强引用Animal对象.这就意味着Animal对象的释放还要依赖于block有没有释放,更直白一点就是Animal对象本来可以更早释放的,但是由于block还强引用了它,导致没有被释放.
小问题:如果一个block截获了一个strong类型的指针变量,那么该指针指向的对象什么时候才会被block释放?是block执行完后就被释放还是要等到block被释放后它才会被释放?
答案是选B.这是因为block虽然执行完了,但是block截获的指针变量依然存活(区别于block代码块里定义的局部变量),因此指针指向的对象依然被强引用.要直到block被释放后该对象才会被释放.
是时候比较一下方法二和方法三了.方法三是方法二的改进型.可以看到方法三中,block执行完成后就将截获的指针变量赋值为nil了,从而指向的对象会被立即释放.这是方法二所做不到的.是不是方法三就等价于方法一了呢?显然不等价.要想等价于方法一,要做的事情只有一个:确保该block一定被调用.否则方法三将等价于方法二.
综上,方法一是最佳实践不会引起任何副作用.是不是方法二就不能用了呢?当然不是,只要你能够确信不会引起其他引用循环(ps:在某种情况下方法二会导致引用循环,及在某种情况下方法二即使没有引起引用循环也导致”截获的对象”不能被释放)自然是可以使用的,我就经常使用,因为它写法更简单.