__strong和__weak的用处

__strong和__weak的用处

背景

iOS日常开发中,当我们用到block的时候,常常需要将变量声明为__weak,用来防止循环引用。但是在block内部,为什么要再声明为__strong__weak还有什么其他的用处吗?

示例

Don't BB, show me the code. 作为一名程序猿,还是直接上代码比较直观,便于理解。

  • eg1,单纯使用__weak的缺点:
@interface MyObject : NSObject
@end

@implementation MyObject
- (void)dealloc {
    NSLog(@"MyObject dealloc");
}
@end


- (IBAction)buttonClickedWeak:(id)sender {
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        MyObject *obj = [[MyObject alloc] init];
        
        __weak typeof(obj) weakObj = obj;
        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            NSLog(@"GCD--sleepBefore obj:%@",weakObj);
            sleep(3);
            NSLog(@"GCD--sleepAfter obj:%@",weakObj);
        });
        sleep(1);
        obj = nil;
        NSLog(@"已经设为nil");
    });
}

如上所示代码,各位看官可以先分析一下,控制台输出的log应该是什么样的,看看和实际结果是否一样。

下面是实际的log输出

 GCD--sleepBefore obj:<MyObject: 0x60c00000eed0>
 MyObject dealloc
 已经设为nil
 GCD--sleepAfter obj:(null)

这里我们把第一个并行队列叫作队列 A ,第二个为队列 B ,显然,当队列 A sleep时,队列 B 已经执行到第一行log,所以这里GCD--sleepBefore的log 可以正常打印出obj。跟着队列 A 被唤醒,obj被置为nil,被释放。所以队列 B 3秒唤醒后,GCD--sleepAfter的log打印的obj为null

  • eg2,联合使用__weak__strong
- (IBAction)buttonclicked2:(id)sender {
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        MyObject *obj = [[MyObject alloc] init];
        
        __weak typeof(obj) weakObj = obj;
        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            __strong typeof(weakObj) strongObj = weakObj; //注意和上面例子的对比区别,这里使用了__strong 修饰
            sleep(3);
            NSLog(@"GCD--obj:%@",strongObj);
        });
        obj = nil; //注意和上面例子的对比区别,这里没有sleep(1)
        NSLog(@"已经设为nil");
    });
}

这次我在block内部强引用了obj,是否可以在队列 B 3s被唤醒后正确的输出呢?

下面是实际的log输出

 MyObject dealloc
 已经设为nil
 GCD--obj:(null)

我们看到,依然不能正确输出,这是因为,block内部声明为__strong只能保证对象在block内部会被强引用,也就是在block执行过程中不会被释放,但是如果在执行到block的时候,对象已经被释放,声明__strong也并不能解决这个问题。

上面所示代码,因为block引用到的外部变量weakObj被声明为__weak,所以block不会强引用obj,所以obj被赋值为nil,会立马触发析构函数,释放obj。所以当执行到block内部的__strong typeof(weakObj) strongObj = weakObj;这条语句时,实际上相当于__strong typeof(weakObj) strongObj = nil;

  • eg3,联合使用__weak__strong
- (IBAction)buttonclicked:(id)sender {
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        MyObject *obj = [[MyObject alloc] init];
        __weak typeof(obj) weakObj = obj;
        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            __strong typeof(weakObj) strongObj = weakObj;
            sleep(3);
            NSLog(@"GCD--obj:%@",strongObj);
        });
        sleep(1); //注意和上面例子的对比区别,这里重新加回 sleep(1)
        obj = nil;
        NSLog(@"已经设为nil");
    });
}

这次我在 obj 赋值nil之前,使队列 A sleep1秒,保证进入队列 B 的时候,obj没有被释放,是否可以在队列 B 3s被唤醒后正确的输出呢?

下面是实际的log输出

 已经设为nil
 GCD--obj:<MyObject: 0x60800000f080>
 MyObject dealloc

这次被正常的输出了,这是因为obj 赋值nil之前,已经执行了对列 B 中的__strong声明,此时obj会被block强引用,当obj 在外面被赋值nil后,并不会触发析构函数,而是在block走完后触发析构。

  • eg4,不使用__weak__strong
//模拟日常工作中,请求接口刷新数据,刷新UI的场景。这里模拟接口已经发出,但是还没相应时,页面已退出的情况。

// 新加一个类,模拟数据管理层,用来管理数据源,刷新UI等
@interface MyDataManager : NSObject
- (void)refresh;
@end
@implementation MyDataManager
- (void)dealloc {
    NSLog(@"MyDataManager dealloc: %@", [NSThread currentThread]);
}
- (void)refresh {
    NSLog(@"refresh with data");
}
@end

- (IBAction)buttonclicked3:(id)sender {
    //创建一个新的DataManager,模拟我们日常工作中的数据层
    MyDataManager *dm = [[MyDataManager alloc] init];

    // 创建一个请求,由于和self、dm等没有引用关系,所以这里不使用__weak修饰,不会有循环引用的问题。
    MyObject *request = [[MyObject alloc] init];
    request.completion = ^{
        //模拟接口请求回来后,刷新数据逻辑
        [dm refresh];
    };
    
    //模拟发送异步请求,3s后收到response
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_global_queue(0, 0), ^{
        // do something parse data,例如JSON解析
        
        //模拟异步主线程,回调完成结果
        dispatch_async(dispatch_get_main_queue(), ^{
            request.completion();
            
            //模拟结束后,异步线程做一些性能统计等事情。
            dispatch_async(dispatch_get_global_queue(0, 0), ^{
                [request doSomething];
            });
        });
    });
    
    //模拟页面请求,接口完成前退出页面,需要销毁的情况
    NSLog(@"finish, dm should dealloc");
}

这里在request.completionblock赋值的时候,由于没有循环引用的问题,所以没有添加__weak修饰。这段代码,是否可以在最后正常释放dm呢?

下面是实际的log输出

 finish, dm should dealloc
 refresh with data
 do something
 MyObject dealloc
 MyDataManager dealloc: <NSThread: 0x600001e57a80>{number = 6, name = (null)}

从日志可以看到,这里有2个问题。

  1. dm的生命周期被延长了,并不能在函数结束后立马释放。

  2. dm的dealloc不是在主线程调用的。

这是因为有如下引用链 dm << request.completion << request << dispatch_after.block \ dispatch_async(dispatch_get_main_queue().block \ dispatch_async(dispatch_get_global_queue(0, 0).block,当最后的异步统计线程block执行完毕后,request会被释放,且是在当前异步线程。然后依次在当前异步线程,触发request.completion、dm的释放。

结论

  1. 在没有循环引用问题的情况下,不使用__weak,一样可能会延长被捕获变量的生命周期,且影响dealloc的线程。UI控件系统在底层重写了dealloc,会自动异步到主线程。这种延迟生命周期的情况,有时正是我们想要的,有时则是超出预期的,所以小伙伴在处理的时候需要注意。
  2. 在需要__weak的情况下,单纯使用__weak,有可能造成block中的变量,在执行过程中被释放,导致代码上逻辑错误。所以推荐在block内部使用__strong。当然,在block内部使用__strong也只能保证block内部执行过程中不会释放对应的对象,并不能保证该对象在执行到block时没有被释放。

最后附上demo,小伙伴可以下载代码进行测试。

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

推荐阅读更多精彩内容