引言
相信大家对block
的使用都不会感到陌生, __weak
和__strong
也成为了大家解决循环引用的利器. 不过大家有否想过, block
出现嵌套情况时, __weak
和__strong
还能好使吗, 或者说会不会有一些需要注意的地方呢?
正篇
首先我们定义了一个Person
类, .h
中只有1个无参无返回的block
属性 -> personBlock
, 并且重写了dealloc
方法, 如下图所示:
接下来我们在主方法中创建一个Person
对象, 由于没有把对象强引用起来, 所以对象一创建出来就被释放了:
接着我们对对象进行会造成循环引用的操作. 由于对象强引用着block
, block
也强引用着对象, 所以控制台并没有出现任何的打印, 编译器也发出了相应的警告:
为了解决循环引用问题, 我们请出了2大神器__weak
和__strong
, 问题当然被迎刃而解, 这里的警告只是因为结果没被使用. 当然了, 此处其实并不需要用到__strong
, 但实际开发中情况复杂多变, 基于普遍适用性的需求, 此处也用上__strong
:
接下来主角出场了, 如果block
中再次对block
赋值的情况下会怎么样呢? 也就是block
嵌套. 我们可能会这样想, strongPerson
已经是可以放心使用的对象了, 还需要担心什么循环引用, 直接套进去用就行了. 真是这样的吗?
果然不出所料? 先别开心太早, 虽然控制台有输出, 对象得到了释放, 可是你有看到编译器已经发出警告了吗? 可能你觉得编译器出毛病了, 乱警告, 其实不然, 这里是一个语法问题, 第二层block
并没有真正被加载进内存. 什么? 不相信? 我只要加一句代码问题就会马上暴露:
只要在最后把外层的block
执行一次, 内层的block
才会真正地被加载进内存, 循环引用问题再次出现. 出现循环引用的原因其实也不难理解, 因为strongPerson
说白了也是一个强引用, 它与一般强引用的区别在于, 它只会在被定义的block
中对对象进行强引用, 在block
过后就会把对象释放掉, 所以在第2层block
中继续用strongPerson
出现循环引用跟一般造成循环引用的原因其实是一样的, 解决方法也是如出一辙, 而且可以继续嵌套下去, 此处的警告同样是结果没被使用:
补充
另外想补充一个关于block
强引用对象的问题, 之前在网上看到过一些说法, 大概就是block
会对内部所有的对象产生一个强引用, 比如说:
self.block = ^{
self.name = @"Veeco";
};
这里block
首先对self
产生一个强引用, 其实还对name
这个属性产生了一个强引用, 在这个补充里, 我就是想要澄清关于这个name
属性被强引用的问题.
首先我们在刚才代码的基础上加一个Dog
类, 只重写其中的dealloc
方法:
接着在Person
类.h
中增加一个Dog
属性 -> dog
, 需要注意的是此处用的是weak
关键字:
正篇开始, 我们在主方法中分别创建person
和dog
对象, 并把dog
对象赋值给person
对象的dog
属性, 接着在person
对象的block
属性中访问person
对象的dog
属性, 按照网上的说法, 如果block
强引用着person
对象的dog
属性的话, 我们是看不到Dog dealloc
打印的, 接下来就来见证奇迹的时刻:
结果出来了, Person
类没被释放这是肯定的, 因为造成了循环引用, 不过我们看到了Dog dealloc
打印, 也就是说block
并没有强引用着person
对象中的dog
属性, 它只强引用着person
对象.