zoukankan      html  css  js  c++  java
  • block(六)循环引用-b

    在ARC与非ARC环境下对block使用不当都会引起循环引用问题,一般表现为,某个类将block作为自己的属性变量,然后该类在block的方法体里面又使用了该类本身,简单说就是self.theBlock = ^(void){[self dosomething];或者self.otherVar = XXX;或者_otherVar = ...};block的这种循环引用会被编译器捕捉到并及时提醒,那要如何避免呢,我在这里做了个demo测试。


    直接看demo
    #import "ViewController.h"
    #define CJSUPARC 1
    typedef void(^myBlock)(void);

      @interface ViewController ()
      @property (nonatomic, copy)myBlock theBlock;
      @end
    
      @implementation ViewController
    
      - (void)dealloc
      {
      #if CJSUPARC
          [super dealloc];
      #else
    
      #endif
          NSLog(@"ViewController dealloc");
      }
    
      - (void)viewDidLoad {
          [super viewDidLoad];
    
      #if CJSUPARC
          __block typeof(self) wSelf = self;
          NSLog(@"%@",@(self.retainCount));
          myBlock block = ^(void){
              NSLog(@"%@",@(wSelf.retainCount));
              NSLog(@"%@.block init",[wSelf.class description]);
          };
          self.theBlock = [[block copy] autorelease];
    
          if (self.theBlock) {
              self.theBlock();
          }
      #else
      //    __block typeof(self) wSelf = self;
      //    __unsafe_unretained typeof(self) wSelf = self;
          __weak typeof(self) wSelf = self;
          myBlock block = ^(void){
              NSLog(@"%@.block init",[wSelf.class description]);
          };
          self.theBlock = [block copy];
    
          if (self.theBlock) {
              self.theBlock();
          }
      #endif
    
      }
      @end
    

    • MRC模式下运行
      2016-01-09 23:50:33.993 CJBlockDemo[52536:2193676] self.retainCount = 6
      2016-01-09 23:50:33.994 CJBlockDemo[52536:2193676] self.retainCount = 6
      2016-01-09 23:50:33.994 CJBlockDemo[52536:2193676] ViewController.block init
      2016-01-09 23:50:40.437 CJBlockDemo[52536:2193676] ViewController dealloc
      可以看到使用__block能够避免引起循环引用的问题

    • ARC模式下运行

    使用__block

      2016-01-09 23:50:33.994 CJBlockDemo[52536:2193676] ViewController.block init
     虽然编译器警告是没有了,但ViewController却没有执行dealloc函数,说明在arc中__block还是会引起retain。
    

    使用__unsafe_unretained

    使用__weak

      2016-01-09 23:50:33.994 CJBlockDemo[52536:2193676] ViewController.block init
      2016-01-09 23:50:40.437 CJBlockDemo[52536:2193676] ViewController dealloc
    

    都可以避免循环引用的问题,但由于前者是unsafe的,会造成野指针问题,所以尽量少用unsafe_unretained关键字


    另外在多线程环境下(block中的wSelf有可能被析构的情况下),需要先将self转为strong指针,避免在运行到某个关键步骤时self对象被析构。
    可参考AFNetworking代码:

    __weak __typeof(self)weakSelf = self;
    AFNetworkReachabilityStatusBlock callback =  ^(AFNetworkReachabilityStatus status) { 
        __strong __typeof(weakSelf)strongSelf = weakSelf;
        strongSelf.networkReachabilityStatus = status;
        if (strongSelf.networkReachabilityStatusBlock) {
             strongSelf.networkReachabilityStatusBlock(status);
        }
    };
    
    • 第一行:__weak __typeof(self)weakSelf = self;
      为防止callback内部对self强引用,weak一下。
      其中用到了__typeof(self),这里涉及几个知识点:
    1. __typeof、__typeof__、typeof的区别
      恩~~他们没有区别,但是这牵扯一段往事,在早期C语言中没有typeof这个关键字,__typeof、__typeof__是在C语言的扩展关键字的时候出现的。
      typeof是现代GNU C++的关键字,从Objective-C的根源说,他其实来自于C语言,所以AFNetworking使用了继承自C的关键字。
    2. 对于老的LLVM编译器上面这句话会编译报错,所以在很早的ARC使用者中流行__typeof(&*self)这种写法,原因如下
      大致说法是老LLVM编译器会将__typeof转义为 XXX类名 *const __strong的__strong和前面的__weak关键字对指针的修饰又冲突了,所以加上&*对指针的修饰。
    • 第三行:__strong __typeof(weakSelf)strongSelf = weakSelf;
      self转回strong了,这里__typeof()里面写的是weakSelf,里面写self也没有问题,因为typeof是编译时确定变量类型,所以这里写self 不会被循环引用。

    • 第四、五、六行,如果不转成strongSelf而使用weakSelf,后面几句话中,有可能在第四句执行之后self的对象可能被析构掉,然后后面的StausBlock没有执行,导致逻辑错误。

    • 最后第五行,使用前对block判空。



    原文链接:https://www.jianshu.com/p/a87ed8a4363a


    =========================看法分割线===============

    是否所有的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,但是这次的循环引用并没有提示,而且也确实发生了 --> 这才是真正告诉面试官:我们做过有实际开发,并且是在真实的开发环境中遇到了-->真正的循环引用!!

    解决办法-->weakSelf!

    参考链接   http://www.jb51.net/article/108171.htm  

  • 相关阅读:
    markdown使用
    clientX、offsetX、screenX、pageX、x的区别
    VMware workstation10 安装CentOS-7-x86_64-DVD-1810
    继承模式、命名空间、对象枚举
    Mac在github中管理自己的代码(入门篇)
    Xcode中常见的错误,警告和解决方法
    Xcode5 添加一个Github/Repository 并且Checkout
    iOS 开发常用网站
    CALayer动画 和 CABasicAnimation用法
    OC属性的setter和getter方法
  • 原文地址:https://www.cnblogs.com/isItOk/p/8108495.html
Copyright © 2011-2022 走看看