iOS block嵌套强弱引用转换 --心得

一、根据需求提出问题

  • 这里直接用一个需求来探究循环引用的问题:如果我想在Block中延时来运行某段代码,这里就会出现一个问题,看这段代码:
    <code class="objectivec">- (<span class="hljs-keyword" style="color: rgb(0, 0, 136);">void</span>)viewDidLoad {
      [<span class="hljs-keyword" style="color: rgb(0, 0, 136);">super</span> viewDidLoad];
      MitPerson*person = [[MitPerson alloc]init];
      __<span class="hljs-keyword" style="color: rgb(0, 0, 136);">weak</span> MitPerson * weakPerson = person;
      person<span class="hljs-variable" style="color: rgb(102, 0, 102);">.mitBlock</span> = ^{
          dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(<span class="hljs-number" style="color: rgb(0, 102, 102);">2</span> * <span class="hljs-built_in" style="color: rgb(102, 0, 102);">NSEC_PER_SEC</span>)), dispatch_get_main_queue(), ^{
              [weakPerson test];
          });
      };
      person<span class="hljs-variable" style="color: rgb(102, 0, 102);">.mitBlock</span>();
    }</code>
    直接运行这段代码会发现[weakPerson test];并没有执行,打印一下会发现,weakPerson 已经是 Nil 了,这是由于当我们的 viewDidLoad 方法运行结束,由于是局部变量,无论是 MitPerson 和 weakPerson 都会被释放掉,那么这个时候在 Block 中就无法拿到正真的 person 内容了。
  • 按如下方法修改代码:
    <code class="objectivec">- (<span class="hljs-keyword" style="color: rgb(0, 0, 136);">void</span>)viewDidLoad {
      [<span class="hljs-keyword" style="color: rgb(0, 0, 136);">super</span> viewDidLoad];
      MitPerson*person = [[MitPerson alloc]init];
      __<span class="hljs-keyword" style="color: rgb(0, 0, 136);">weak</span> MitPerson * weakPerson = person;
      person<span class="hljs-variable" style="color: rgb(102, 0, 102);">.mitBlock</span> = ^{
          __<span class="hljs-keyword" style="color: rgb(0, 0, 136);">strong</span> MitPerson * strongPerson = weakPerson;
          dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(<span class="hljs-number" style="color: rgb(0, 102, 102);">2</span> * <span class="hljs-built_in" style="color: rgb(102, 0, 102);">NSEC_PER_SEC</span>)), dispatch_get_main_queue(), ^{
              [strongPerson test];
          });
      };
      person<span class="hljs-variable" style="color: rgb(102, 0, 102);">.mitBlock</span>();
    }</code>
    这样当2秒过后,计时器依然能够拿到想要的 person 对象。
  • 二、深入探究原理

    • 这里将会对每行代码逐步进行说明
      <code class="cpp"><span class="hljs-number" style="color: rgb(0, 102, 102);">1</span>、开辟一段控件存储 person 类对象内容,创建 person 强指针。
      MitPerson*person = [[MitPerson alloc]init];</code>
      <code class="objectivec"><span class="hljs-number" style="color: rgb(0, 102, 102);">2</span>、创建一个弱指针 weakPerson 指向person对象内容 
      __<span class="hljs-keyword" style="color: rgb(0, 0, 136);">weak</span> MitPerson * weakPerson = person;</code>
      <code class="cpp">person.mitBlock = ^{
      <span class="hljs-number" style="color: rgb(0, 102, 102);">3</span>、在 person 对象的 Block 内部创建一个强指针来指向 person 对象,为了保证当计时器执行代码的时候,person 对象没有被系统销毁所以我们必须在系统内部进行一次强引用,并用 GCD 计时器引用 strongPerson,为了保留 person 对象,在下面会对这里更加详细的说明。
        __strong MitPerson * strongPerson = weakPerson;
          dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (<span class="hljs-keyword" style="color: rgb(0, 0, 136);">int64_t</span>)(<span class="hljs-number" style="color: rgb(0, 102, 102);">2</span> * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
              [strongPerson test];
          });
      };</code>
      <code class="mathematica"><span class="hljs-number" style="color: rgb(0, 102, 102);">4</span>、执行 <span class="hljs-keyword" style="color: rgb(0, 0, 136);">Block</span> 代码
        person.mitBlock();</code>
    • 下面将详细分析一下下面这段代码:
      <code class="objectivec">person<span class="hljs-variable" style="color: rgb(102, 0, 102);">.mitBlock</span> = ^{
      __<span class="hljs-keyword" style="color: rgb(0, 0, 136);">strong</span> MitPerson * strongPerson = weakPerson;
          dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(<span class="hljs-number" style="color: rgb(0, 102, 102);">2</span> * <span class="hljs-built_in" style="color: rgb(102, 0, 102);">NSEC_PER_SEC</span>)), dispatch_get_main_queue(), ^{
              [strongPerson test];
          });
      };</code>
    • 首先需要明白一些关于 Block 的概念:
      • 1、默认情况下,block 是放在栈里面的
      • 2、一旦blcok进行了copy操作,block的内存就会被放在堆里面
      • 3、堆立面的block(被copy过的block)有以下现象
        • 1> block内部如果通过外面声明的强引用来使用,那么block内部会自动产生一个强引用指向所使用的对象。
        • 2> block内部如果通过外面声明的弱引用来使用,那么block内部会自动产生一个弱引用指向所使用的对象。
    • 我们进行这段代码的目的:
      • 首先,我们需要在 Block 块中调用,person 对象的方法,既然是在 Block 块中我们就应该使用弱指针来引用外部变量,以此来避免循环引用。但是又会出现问题,什么问题呢?就是当我计时器要执行方法的时候,发现对象已经被释放了。
      • 接下来就是为了避免 person 对象在计时器执行的时候被释放掉:那么为什么 person 对象会被释放掉呢?因为无论我们的person强指针还是 weakPerson 弱指针都是局部变量,当执行完ViewDidLoad 的时候,指针会被销毁。对象只有被强指针引用的时候才不会被销毁,而我们如果直接引用外部的强指针对象又会产生循环引用,这个时候我们就用了一个巧妙的代码来完成这个需求。
      • 首先在 person.mitBlock 引用外部 weakPerson,并在内部创建一个强指针去指向 person 对象,因为在内部声明变量,Block 是不会强引用这个对象的,这也就在避免的 person.mitBlock 循环引用风险的同时,又创建出了一个强指针指向对象。
      • 之后再用 GCD 延时器 Block 来引用相对于它来说是外部的变量 strongPerson ,这时延时器 Block 会默认创建出来一个强引用来引用 person 对象,当 person.mitBlock 作用域结束之后 strongPerson 会跟着被销毁,内存中就仅剩下了 延时器 Block 强引用着 person 对象,2秒之后触发 test 方法,GCD Block 内部方法执行完毕之后,延时器和对象都被销毁,这样就完美实现了我们的需求。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值