实际开发中-Block导致循环引用的问题

作者: 小蠢驴打代码 | 来源:发表于2017-03-09 17:46 被阅读1998次

说到循环引用问题,最最最常遇到的,不是在项目中,而是在面试中。如果面试官问你开发中是否遇到过retain cycle,你如果说没遇到过,估计已经很难跟面试官继续友好的沟通下去了。

但是这个问题怎么回答呢,网络上千篇一律的答案-->使用Block的时候遇到过,使用__weakSelf 代替 self 等等,可以说这个答案没啥错,但是所有人都回答的一样,并不能突出我们的逼格,无法让面试官知道我们在这方面有过研究,有闪光点。

对于开发者来说,喜欢探索,喜欢挖掘不懂的知识,在面试官眼里会加分不少。探索是基于问题之上的-->比如:是否所有的Block中,使用self 都会导致循环引用?


系统自带Block不会发生循环引用

如图,使用系统自带的UIView 的Blcok,控制器能被销毁-->说明没有发送循环引用。

原理: UIView的调用的是类方法,当前控制器不可能强引用一个类 ,所以循环无法形成 --> 动画block不会造成循环引用的原因。


所以通过实践得出第一个结论--> 并不是所有的Block中使用self,都会导致循环引用!


问题二:面试官问:那除了系统自带的方法中的Block,你在其他Block中使用self 会导致循环引用吗? -->可答:AFN框架!

最常用的数据请求框架-- AFNetWorking框架的Block是否会强引用?

AFN的Block是否会导致循环引用测试

如上图所示,在AFN的 block { xxx self.view  } 使用self,并不会导致循环引用!

原理:AFN无循环是因为绝大部分情况下,你的网络类对象是不会被当前控制器引用的,这时就不会形成引用环。(查阅资料得知)

小tips:也可能AFN底层有自己做了操作,这里没探究到AFN框架底层,仅知道AFN不会造成循环引用。


那什么情况下会导致循环引用呢? --> 自定义Block

自定义Block中使用self

添加 viewDidLoad 提示框-->每次进入都打印viewDidLoad,可以确定是否离开视图控制器-->如果是,但是没有调用dealloc --> 循环引用

循环引用

这时候,我们发现循环引用发生了!所有我们答道:“我们在实际开发中,使用自定义Block,在Block { xxx }中使用self,导致了循环引用 ”

循环引用导致的原因: 相互强指向

循环引用原因

如何解决-->使用weakSelf,这个解决方法估计没见过一百次的,都不算是真正参加过iOS面试的。

----------------------------- 华丽分割线--------------------------------------

一个大写的excuse me 写脸上,49行都报警告了,而且提示可能发送循环引用,这你都能因为这样导致循环引用??这面试官如果知道这个,应该不会这么友好的放过你吧?


由于现在学iOS的太多了,所有可能面试官如果对于循环引用比较了解的话,并不会因为我们回答了上面两个问题就放过我们~他可能会接着问:那如果是我们自己写的Block,(非系统和AFN),在Block中使用self,是否一定会发生循环引用~


探究四:自定义Block是否一定会发生循环引用?

在其他控制器声明一个强指向的Block 调用Blcok 执行效果

如图:发现oneVC被销毁了,说明,自己定义的Block,里面使用了self,并不一定会发生循环引用!

原理:block --> 强指向了self,但是self,并没有指向Block!-->并没有一个 self.block 或者 成员变量 @property block ,所有Block并没有被强指向-->没有发送循环引用!

-->Tips:循环引用发生的条件就是持有这个block的对象,被block里边加入的对象持有。


逼格出现了!!华丽分割线! 既然系统的Block、AFN、都不会发生循环引用,自定义Block又有这么明显的提示-->实际开发中不会遇到循环引用?? 

---------------------------------高逼格分割线-----------------------------------------

实际开发中:使用通知(NSNotifation),调用系统自带的Block,在Block中使用self --> 会发生循环引用。

通知的接收方法

现在iOS的通知已经比较好用了,如图第二个方法,我最常用的,特别方便,不需要写@selector(方法)+ 调用,直接写在Block中,就可以实现接收通知之后实现的代码。

twoVC发送通知 --> 给oneVC oneVC 接收通知 使用通知-发生循环引用

如图!这才是实际开发中-->真正有可能发生循环引用的地方!确实也是在通知的Block,但是这次的循环引用并没有提示,而且也确实发生了 --> 这才是真正告诉面试官:我们做过有实际开发,并且是在真实的开发环境中遇到了-->真正的循环引用!!(不仅仅是面试题讲的一个Block的事,逼格明显不够)

解决办法-->weakSelf!



相关文章

网友评论

  • Harely:满满的干货,很不错有源代码就更好了
    Harely:@小蠢驴打代码现在有demo的习惯吗
    小蠢驴打代码:之前写的没有顺便贴demo的习惯,也没备份到git上,找不到了了:joy:
  • louyw:在AFN中为什么不会出现循环引用,原因很简单,不用看底层,因为self不持有AFN的block,只是AFN的block持有self,仅仅是单向持有,并没有相互强引用~所以不会造成循环引用,就像你在oneVC中使用twoVC里面定义blcok不会出现循环引用一样,UIView动画也一个道理~
  • PRLiu:我想问下,修饰block属性的时候,一定要strong和copy么,如果是assign是否可以?我用assign,是没有造成循环引用的。求指教!!!
    PRLiu:@property(nonatomic,assign)NSString*property_assign;
    @property(nonatomic,copy)NSString*property_copy

    //assign是简单的赋值
    NSString *temp_str = @"1";
    NSLog(@"temp_str:%p----%ld",temp_str,CFGetRetainCount((__bridge CFTypeRef)(temp_str)));

    self.property_assign = temp_str;
    NSLog(@"property_assign:%p----%ld",self.property_assign,CFGetRetainCount((__bridge CFTypeRef)(self.property_assign)));
    self.property_assign = temp_str;
    NSLog(@"property_assign:%p----%ld",self.property_assign,CFGetRetainCount((__bridge CFTypeRef)(self.property_assign)));

    //copy是开释旧成员变量,并新分派内存地址给成员变 量,将传入参数内容复制一份,给成员变量
    self.property_copy = [temp_str copy];
    NSLog(@"property_copy:%p----%ld",self.property_copy,CFGetRetainCount((__bridge CFTypeRef)(self.property_copy)));
    self.property_copy = temp_str;
    NSLog(@"property_copy:%p----%ld",self.property_copy,CFGetRetainCount((__bridge CFTypeRef)(self.property_copy)));


    2017-03-24 09:03:14.911 Test_ everyday[85739:3853303] temp_str:0x1025a54a0----1152921504606846975
    2017-03-24 09:03:14.911 Test_ everyday[85739:3853303] property_assign:0x1025a54a0----1152921504606846975
    2017-03-24 09:03:14.912 Test_ everyday[85739:3853303] property_assign:0x1025a54a0----1152921504606846975
    2017-03-24 09:03:14.912 Test_ everyday[85739:3853303] property_copy:0x1025a54a0----1152921504606846975
    2017-03-24 09:03:14.912 Test_ everyday[85739:3853303] property_copy:0x1025a54a0----1152921504606846975

    请问 我为何实践的时候 发现copy的作用跟博客说的不一样的?
    AirSars:block创建是在stack里的,如果你block里引用了self,并且要执行self接下来的代码,那你必须得用strong或者copy(block的strong操作其实也是copy)来复制一份到heap里,不然self会被释放从而造成crash错误,如果没有接下来的其他操作,你用assign声明也是可以的
    小蠢驴打代码:简单的来说,就是强弱引用的问题,如果 block 用 strong 或者 copy 修饰,就说明是当前控制器强引用,循环引用发生的前提是 强引用相互指向,如果你使用弱引用--assign(或者weak),自然地就避免了循环引用的发生。(建议:看一下上图的“循环引用原因”,如果一方是assign,可以将线换成虚线)
  • b50eacfb59f6:总结的非常有针对性,完整解答了block循环引用的相关问题。逼格很高,兄弟干的漂亮!
    小蠢驴打代码:兄弟,你这围观的水军也太耿直了一点吧
  • 帅气的小时:作者666,写的很仔细诶
    小蠢驴打代码:@帅气的小时 排版丑了点:joy:不认真看的估计都看不懂写了啥
  • Cooci_和谐学习_不急不躁:作者整理挺用心的,加油,我持续关注你
    小蠢驴打代码:@西单_夜未央 谢谢!你这波眼光还是可以的:baby_chick:

本文标题:实际开发中-Block导致循环引用的问题

本文链接:https://www.haomeiwen.com/subject/pvzwgttx.html