block-循环引用

block的循环引用,在日常开发中,我们常常遇到,但是可能部分新人还不太了解为何会循环引用,到底是如何循环引用理解得不够透彻,并且在ARC环境下只知道用__weakSelf去解决,但也不知道原因,现在我们来剖析一下,循环引用的的底层原理。看看下面一段常见的代码:

循环引用原因分析

#import <Foundation/Foundation.h>
#import "RMPerson.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        RMPerson *person = [[RMPerson alloc] init];
        person.age = 20;
        person.block = ^{
            NSLog(@"age is %d",person.age);
        };
        person.block();
    }
    NSLog(@"----------------");
    return 0;
}

---------------RMPerson.h----------------
#import <Foundation/Foundation.h>
@interface RMPerson : NSObject
@property (nonatomic, assign) int age;
@property (nonatomic, copy) void (^block)(void);
@end
---------------RMPerson.h----------------
#import "RMPerson.h"
@implementation RMPerson
@end

// 控制台打印
2018-07-04 11:27:23.129229+0800 block-循环引用[26623:2507647] age is 20
2018-07-04 11:27:23.129421+0800 block-循环引用[26623:2507647] -----------------
Program ended with exit code: 0

从上面的代码可以得出,block调用完后,person都没用释放,NSLog(@"----------------");打印完了,person也还没释放,说明person引用计数器不为0。
用clang 命令将 这段代码转换成C++代码(xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m)如下:

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  RMPerson *__strong person;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, RMPerson *__strong _person, int flags=0) : person(_person) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  RMPerson *__strong person = __cself->person; // bound by copy

            NSLog((NSString *)&__NSConstantStringImpl__var_folders_5l_0xn052bn6dgb9z7pfk8bbg740000gn_T_main_d61985_mi_0,((int (*)(id, SEL))(void *)objc_msgSend)((id)person, sel_registerName("age")));
        }

static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {

_Block_object_assign((void*)&dst->person, (void*)src->person, 3/*BLOCK_FIELD_IS_OBJECT*/);

}

static void __main_block_dispose_0(struct __main_block_impl_0*src) {
    
    _Block_object_dispose((void*)src->person, 3/*BLOCK_FIELD_IS_OBJECT*/);

}

static struct __main_block_desc_0 {
  size_t reserved;
  size_t Block_size;
  void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*);
  void (*dispose)(struct __main_block_impl_0*);
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0), __main_block_copy_0, __main_block_dispose_0};

int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 


        RMPerson *person = ((RMPerson *(*)(id, SEL))(void *)objc_msgSend)((id)((RMPerson *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("RMPerson"), sel_registerName("alloc")), sel_registerName("init"));
        ((void (*)(id, SEL, void (*)()))(void *)objc_msgSend)((id)person, sel_registerName("setBlock:"), ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, person, 570425344)));
        ((void (*(*)(id, SEL))())(void *)objc_msgSend)((id)person, sel_registerName("block"))();
    }
    return 0;
}
static struct IMAGE_INFO { unsigned version; unsigned flag; } _OBJC_IMAGE_INFO = { 0, 2 };

这段代码,我们再熟悉不过了,因为从之前block的章节中,几乎每一个相关的block小问题,都会上底层代码来研究其原理。循环引用的原因就是,1.person里的属性block强引用block,因block又捕获了person,block在copy的时候,_Block_object_assign根据person是strong,还是__weak,来对person的引用计数器做是否+1的处理,而此处的person是strong修饰的,所以block又强引用person,person的引用计数器+1。所以2.block又强引用了person
用图片表示就是下面:

block循环引用

解决循环引用问题

1、用__weak、__unsafe_unretained解决
//  __weak
RMPerson *person = [[RMPerson alloc] init];
person.age = 20;
__weak typeof(person) weakPerson = person;
person.block = ^{
    NSLog(@"age is %d",weakPerson.age);
};
person.block();
  • 如上使用__weak可解决循环引用,也是开发中最常用最安全的方法,使用__weak,block捕获person的进去时,block不会强引用person,而是对对象弱引用,如下图
    弱引用
// __unsafe_unretained
RMPerson *person = [[RMPerson alloc] init];
person.age = 20;
__unsafe_unretained typeof(person) weakPerson = person;
person.block = ^{
    NSLog(@"age is %d",weakPerson.age);
};
person.block();
  • 使用__unsafe_unretained也可以解决,__unsafe_unretained顾名思义就是不安全、不retained的意思,__unsafe_unretained__weak相比较主要区别是在于,当person释放的时候,block也随之销毁,但是在__unsafe__unretained修饰下的weakPerson会依然指向之前的内存空间,此时weakPerson访问的就是"僵尸对象",所以就是不安全。
总结:
  • __unsafe_unretained: 不会对对象进行retain,当对象销毁时,会依然指向之前的内存空间(野指针)
  • __weak: 不会对对象进行retain,当对象销毁时,会自动指向nil

相比之下,建议开发中使用__weak__weak更安全,更有效

2、用__block解决(必须调用block)
RMPerson *person = [[RMPerson alloc] init];
__block weakPerson = person;
self.block = ^ {
     NSLog(@"person : %@",weakPerson);
     weakPerson = nil;
};
self.block();

__block修饰的变量,会被包装成一个对象,也持有了person对象,如下面一张图


所以,要解决循环引用,就把__block变量持有的person对象的指针置为nil后,就可以解决,也因此必须要调用block,才能将__block变量置空如下图:

总结:循环引用是因为,对象强引用了blcok,block内部也强引用了捕获进去的对象,相互引用无法释放。解决循环引用,1、用__weak、__unsafe_unretained解决,2、用__block解决(必须调用block).


以上分析得解决循环引用的都是在ARC环境下的,现在也简单下分析MRC环境下是如何避免循环引用。
解决循环引用(MRC环境)

1、用__unsafe_unretained解决(MRC环境下是不支持__weak弱指针的)

// __unsafe_unretained
RMPerson *person = [[RMPerson alloc] init];
person.age = 20;
__unsafe_unretained typeof(person) weakPerson = person;
person.block = ^{
    NSLog(@"age is %d",weakPerson.age);
};
person.block();

2、用__block解决

RMPerson *person = [[RMPerson alloc] init];
__block weakPerson = person;
self.block = ^ {
     NSLog(@"person : %@",weakPerson);
};

在MRC环境下,__block不会对person强引用,所以不会存在循环引用。

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

推荐阅读更多精彩内容