从调用 Masonry 的 mas_makeConstraints 方法而其 block 参数中可以不使用 WeakSelf 说说 Block 的 Retain Circle。

标题确实很长,有点长……😆。

不理解标题意思没问题,看看下面的代码就知道这次我想复习的地方是什么了。

UIButton *testButton = [[UIButton alloc] init];
[self.view addSubview:testButton];
[testButton mas_makeConstraints:^(MASConstraintMaker *make) {
    make.width.equalTo(@100);
    make.height.equalTo(@100);
    make.left.equalTo(self.view.mas_left);
    make.top.equalTo(self.view.mas_top);
}];

上面的 block 里面使用了self,这样子不就是循环引用了吗?真的retain circle了吗?。。。然后一阵沉默。。。。
其实面对突如其来的问题,即使不能马上回答,或者真的被别人鄙视了也应该抱着无所谓的的态度(其实心里已经一百万只cnm在奔腾了。。。😝),接下来耗一点点的娱乐时间,把问题从自己理解的角度去思考为什么且一步一步解答下来才能真正将该知识点吃进自己的心里。时至今日,答案这类生物在互联网上还是能够很好地捕捉的到,还没到“拼爹”的地步呢~

进入正题前……,上述代码不会出现 retain circle。


一、什么是 retain circle?

retain cycle表示两个对象之间互相强引用/互相retain对方的情况,导致对象之间谁也释放不了,造成内存泄露。
从reference count 方面解析,即当两个对象互相强引用的时候,双方的引用计数都是+1的,导致任何时候引用计数都不为0,始终无法释放,无法释放他们的内存,即使已经没有变量持有他们。

二、打破 retain circle

其中最常用的方法是:

  1. 使用 weak 弱引用修饰属性
  2. 使用__weak修饰 self
  3. 在 block 调用之后将 block 置 nil。
    (以上情况之后有空再补充说明,相信有经验的你可以大概配对到对应的使用场景了😋)

三、block 中引用 self 要注意循环引用问题

很多时候,我们遇到 block 第一反应都会问,是否需要使用 weakSelf 防止循环引用,这也是苹果文档Avoid Strong Reference Cycles when Capturing self提醒的问题之一。

从以上可以知道,block 会捕获外部变量,并且强引用它。上述例子也是一个经典的 block 循环引用例子。那让我们分析一下文章开头的代码:

UIButton *testButton = [[UIButton alloc] init];
[self.view addSubview:testButton];
[testButton mas_makeConstraints:^(MASConstraintMaker *make) {
    make.width.equalTo(@100);
    make.height.equalTo(@100);
    make.left.equalTo(self.view.mas_left);
    make.top.equalTo(self.view.mas_top);
}];
  1. 控制器 self 会强引用自己的子类 view。self ---> view
  2. 因为testButton实例被添加都 view 内,同样view会强引用testButton。view ---> testButton
  3. testButton调用了 masonry 的 mas_makeConstraints方法,其中参数 block 中引用了 self,问题来了,这里的 block 对 self 是强引用吗?答案是,是的,请留意上面苹果文档中提过locks maintain strong references to any captured objects, including self。那就是说这样使用是会循环引用吗?
  4. 其实,面对第3步的分析,我们还有一个点给忽略了,此刻我们也可以参照上面苹果文档中举的经典例子,该例子中开始的条件是,self copy 了 block 这个属性,它的循环链是:self ---> block ---> self。
    那好,我们回到第3步来,确实,mas_makeConstraints方法,其中参数 block 中强引用了 self,但 testButton 对 block 有强引用吗?现在的循环链情况是:self ---> view ---> testButton ?block ---> self;问号是强还是弱呢?
  5. 要想知道第4步的答案,得进入mas_makeConstraints方法里看看它的实现方式才行:
- (NSArray *)mas_makeConstraints:(void(^)(MASConstraintMaker *))block {
    self.translatesAutoresizingMaskIntoConstraints = NO;
    MASConstraintMaker *constraintMaker = [[MASConstraintMaker alloc] initWithView:self];
    block(constraintMaker);
    return [constraintMaker install];
}

这里的block是局部变量, 也就是说当 block 超出了方法的作用域时就会被销毁,相当于前面的第二标题打破 retain circle 中最后描述的方法在 block 调用之后将 block 置 nil一样。所以,上述的的循环链应该是:self ---> view ---> testButton --x--> block ---> self;并无造成循环引用。

当然,使用第三库时候有任何问题都应该先通过 issue 来排查一下,看看是否早已有人和你一样遇到同样的问题~

总结

综上所述,在使用 masonry 时调用方法 mas_makeConstraints 中可以不使用 weakSelf。

留给自己的作业:

  1. copy 与 strong 有啥区别?都是强引用吗?
  2. strong weak dance 用在哪里?怎样的 weak strong 定义最好?

Ref

  1. 常见循环引用问题产生点:iOS 中的循环引用
  2. 从堆栈角度了解循环引用的产生原因:循环引用,看我就对了
  3. Programming with Objective-C
  4. MRC下 block 的 retain circle:正确使用Block避免Cycle Retain和Crash
  5. block 中捕获的实例变量问题:Block 中何時可以直接用 self,何時必須用 weakSelf / strongSelf?
  6. 带图说明循环引用的:iOS中block的循环引用问题

推荐阅读更多精彩内容