你真的会用strong-weak dance吗?

下文的讨论基于ARC

平时开发中我们遇到block里面引用self的情况,大部分都是这样处理的

__weak typeof(self) weakSelf = self;
self.myBlock =  ^{
  __strong typeof(self) strongSelf = weakSelf;
  [strongSelf doSomething]; 
  [strongSelf doSomethingElse]; 
};

转载请注明出处:来自LeonLei的博客http://www.gaoshilei.com

我们习惯了这样用,貌似这样用了之后可以解决循环引用的问题,而且可以保证block执行之前self不会被释放掉?真相总是残酷的,然而事实并非如此!下面将会对block中引用self的三种方式进行讨论,并给出原因和另外一种解决方案。

1.block中直接引用self

这种情况使用是block被没有被self强引用,因此这样不会导致retain cycle。

dispatch_block_t completionHandler = ^{
    NSLog(@"%@", self);
}

2.在block外部创建weakself变量,在block中引用weakself

当block被self强引用,此时如果在block内强引用self将会导致retain cycle。所以我们就想到了在block外部创建一个weakself,然后block在创建的时候捕获到的是weakself,这样就不会导致retain cycle。

__weak typeof(self) weakSelf = self;
dispatch_block_t block =  ^{
    [weakSelf doSomething]; 
    [weakSelf doSomethingElse]; 
};

但是要注意的是block捕获的是weakself变量,如果在执行doSomething的过程中self被释放掉,由于是弱引用,weakself也将置空,下面的doSomethingElse是无法得到执行的,看一个例子:

下面的例子展示的是,在block调用之后的1秒后释放self,在block中调用doSomething,2秒之后再调用doAnotherThing,意味着调用doAnotherThing之前self已经被释放了

//viewController.m
- (void)viewDidLoad {
    [super viewDidLoad];
    self.sself = [strongweakself new];
    [self.sself test];
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSLog(@"self.block被释放!");
        self.sself = nil;
    });
}

//strongweakself.m
-(void)test
{
    self.myobject = [TestObject new];
    __weak typeof(self) __weakself = self;
    [self.myobject setWeakblock:^{
        NSLog(@"调用block!");
        [__weakself doSomething];
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            [__weakself doAnotherThing];
        });
    }];
    self.myobject.weakblock();
}

-(void)doSomething
{
    NSLog(@"%s",__func__);
}

-(void)doAnotherThing
{
    NSLog(@"%s",__func__);
}

-(void)dealloc{
    NSLog(@"%s",__func__);
}

从打印日志可以看出,block执行大约1秒之后self被dealloc,doAnotherThing并没有得到调用

2017-01-16 14:31:13.834 strong-weak dance[11366:4727954] 调用block!
2017-01-16 14:31:13.836 strong-weak dance[11366:4727954] -[strongweakself doSomething]
2017-01-16 14:31:14.893 strong-weak dance[11366:4727954] self.block被释放!
2017-01-16 14:31:14.893 strong-weak dance[11366:4727954] -[strongweakself dealloc]

所以只使用weakself,在self被释放之后,weakself由于self的释放已经为空,后面的self都将失效,所以在block中这样引用self是非常危险的,下面就要谈谈我们最熟悉的strong-weak dance了。

3.strong-weak dance

对比第二种方案我们看一下doAnotherThing是否可以得到调用,稍微改一下代码,还是在block执行1秒后释放self,我们看看后面的self引用是否有效

-(void)test
{
    self.myobject = [TestObject new];
    __weak typeof(self) __weakself = self;
    [self.myobject setWeakblock:^{
        NSLog(@"调用block!");
        __strong typeof(self) __strongself= __weakself;
        [__strongself doSomething];
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            [__strongself doAnotherThing];
        });
    }];
    self.myobject.weakblock();
}

此时看打印日志:

2017-01-16 14:36:39.039 strong-weak dance[11374:4728878] 调用block!
2017-01-16 14:36:39.039 strong-weak dance[11374:4728878] -[strongweakself doSomething]
2017-01-16 14:36:40.110 strong-weak dance[11374:4728878] self.block被释放!
2017-01-16 14:36:41.213 strong-weak dance[11374:4728878] -[strongweakself doAnotherThing]
2017-01-16 14:36:41.213 strong-weak dance[11374:4728878] -[strongweakself dealloc]

虽然self被释放掉了,但是并没有dealloc,因为block内部的strongself对他进行了一次retain,当doAnotherThing执行完毕,strongself对他的引用计数减一,self被dealloc彻底销毁。

那么问题来了!strong-weak dance能不能解决block执行前,self被释放的问题?下面继续验证
我们改一下代码,在1秒之后释放self,在2秒之后执行block(注意延时block中对于self的处理是weakself,防止延时block对self进行retain影响验证结果

//viewController.m
- (void)viewDidLoad {
    [super viewDidLoad];
    self.sself = [strongweakself new];
    [self.sself test];
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSLog(@"self.block被释放!");
        self.sself = nil;
    });
}

//strongweakself.m
-(void)test
{
    self.myobject = [TestObject new];
    __weak typeof(self) __weakself = self;
    [self.myobject setWeakblock:^{
        NSLog(@"调用block!");
        __strong typeof(self) __strongself= __weakself;
        [__strongself doSomething];
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            [__strongself doAnotherThing];
        });
    }];
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSLog(@"%@",__weakself);
        __weakself.myobject.weakblock();
    });
}

看一下日志:

2017-01-16 14:44:26.314 strong-weak dance[11395:4730727] self.block被释放!
2017-01-16 14:44:26.314 strong-weak dance[11395:4730727] -[strongweakself dealloc]
2017-01-16 14:44:27.372 strong-weak dance[11395:4730727] (null)

当开始调用block的时候报错了,self这时已经被dealloc掉。strong-weak dance并没有解决这种问题。看到这心是不是凉了半截?真相就是如此,我们平时一直使用的strong-weak dance也只能解决block得到调用之后self不被释放的问题。

这是我们最常用的是一种方案,因为block创建时捕获的是weakself,所以block执行之前不能够控制self的生命周期,所以这样不会导致整个block对self进行强引用。之后在block内部创建一个对self进行retain的变量strongself,strongself 作为局部变量强引用了 self 并且会在block执行完毕的时候被自动销毁,这样既可以保证在block执行期间 self 不会被外界干掉,同时也解决了retain cycle的问题。

总结

通过上面几个小栗子可以看出来:strong-weak dance确实是比较好的解决方案,但是也不是万能的,他不能解决block调用之前self被释放的问题,下面将block中引用self分为4中场景:

1. 使用self

当self不持有、不间接持有block时,可以在block内部直接引用self。

2.使用weakself

当self持有或间接持有block,可以通过在外部创建self的弱引用weakself然后捕获到block内部进行使用,但是这样使用存在一定风险,一般也不推荐使用。

3.使用strong-weak dance

当self持有或间接持有block,此时要使用strong-weak dance。
这种方法也不是万能的,在block被执行前,block对self依然只是弱引用,进入block里面才会retain一次,保证在block执行期间self都不会被释放掉。

4. block中强引用self并且打破retain cycle

不管是weakself还是strong-weak dance,目的都是避免retain cycle,strong-weak dance的本质也是在block中搞了一个局部变量来打破这种循环引用的;
如果我们在block中直接使用self,并且在适当的时机打破这种循环(比如说在block执行完成将这个block销毁)也可以避免retain cycle,并且这种在block创建时就强引用的方式,在block被调用前 self 不会被释放掉,可以弥补strong-weak dance的不足。

关于本文的内容可能存在不足的地方,欢迎大家指正!

参考资料

  1. http://albertodebortoli.com/blog/2013/08/03/objective-c-blocks-caveat
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,560评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,104评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,297评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,869评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,275评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,563评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,833评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,543评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,245评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,512评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,011评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,359评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,006评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,062评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,825评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,590评论 2 273
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,501评论 2 268

推荐阅读更多精彩内容