Block由浅入深(6):循环引用

循环引用原因

我们都知道,使用Block的时候不小心就会造成循环引用的问题,那么为什么会出现循环引用呢,本文我们同样使用前文所有的工具和已经得到的原理来分析这个问题。

ARC环境下的循环引用

我们有如下代码,有经验的同学可以一眼就看出来这段代码有循环引用问题:

typedef void(^blk_t)(void);

@interface MyObject : NSObject
{
    blk_t blk;
}

@end

@implementation MyObject

- (instancetype)init {
    if (self = [super init]) {
        blk = ^{NSLog(@"self = %@", self);};
    }

    return self;
}

@end

那么为什么会发生循环引用呢,同样使用clang工具将以上代码转换成C++代码,我们只提取了重要的部分并做了简化来说明:

使用命令clang -rewrite-objc -Wno-deprecated-declarations -fobjc-arc $file.m-fobjc-arc明确将使用ARC。

struct MyObject_IMPL {
    struct NSObject_IMPL NSObject_IVARS;
    __strong blk_t blk;
};

struct __MyObject__init_block_impl_0 {
  struct __block_impl impl;
  struct __MyObject__init_block_desc_0* Desc;
  MyObject *__strong self;
  __MyObject__init_block_impl_0(void *fp, struct __MyObject__init_block_desc_0 *desc, MyObject *__strong _self, int flags=0) : self(_self) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

static instancetype _I_MyObject_init(MyObject * self, SEL _cmd) {
    if (self = [super init]) {
        blk = &__MyObject__init_block_impl_0((void *)__MyObject__init_block_func_0, &__MyObject__init_block_desc_0_DATA, self, 570425344));
    }

    return self;
}

上述代码中,MyObject_IMPL相当于是MyObject的成员变量的实现,__MyObject__init_block_impl_0是代码里block的实现。我们可以看到,MyObject_IMPL中的blk变量是__strong修饰的,所以它会持有blk,同样block的实现中,self也是__strong修饰的,所以blk也会持有self,这样就造成了循环引用。

MRC环境下的循环引用

我们使用同样的例子来说明MRC环境下的循环引用问题,不过为了让代码能够正常运行,需要做一些修改:

typedef void(^blk_t)(void);

@interface MyObject : NSObject
{
    blk_t blk;
}

@end

@implementation MyObject

- (instancetype)init {
    if (self = [super init]) {
        blk = [^{NSLog(@"self = %@", self);} copy];
    }

    return self;
}

- (void)dealloc {
    [super dealloc];
}

@end

我们同样转化为C++代码,主要代码如下:

使用命令clang -rewrite-objc -Wno-deprecated-declarations -fno-objc-arc $file.m-fno-objc-arc指定不使用ARC。

struct MyObject_IMPL {
    struct NSObject_IMPL NSObject_IVARS;
    blk_t blk;
};

struct __MyObject__init_block_impl_0 {
  struct __block_impl impl;
  struct __MyObject__init_block_desc_0* Desc;
  MyObject *self;
  __MyObject__init_block_impl_0(void *fp, struct __MyObject__init_block_desc_0 *desc, MyObject *_self, int flags=0) : self(_self) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

static void __MyObject__init_block_copy_0(struct __MyObject__init_block_impl_0*dst, struct __MyObject__init_block_impl_0*src) {_Block_object_assign((void*)&dst->self, (void*)src->self, 3/*BLOCK_FIELD_IS_OBJECT*/);}

static void __MyObject__init_block_dispose_0(struct __MyObject__init_block_impl_0*src) {_Block_object_dispose((void*)src->self, 3/*BLOCK_FIELD_IS_OBJECT*/);}

static struct __MyObject__init_block_desc_0 {
  size_t reserved;
  size_t Block_size;
  void (*copy)(struct __MyObject__init_block_impl_0*, struct __MyObject__init_block_impl_0*);
  void (*dispose)(struct __MyObject__init_block_impl_0*);
} __MyObject__init_block_desc_0_DATA = { 0, sizeof(struct __MyObject__init_block_impl_0), __MyObject__init_block_copy_0, __MyObject__init_block_dispose_0};

同样,MyObject_IMPL相当于是MyObject的成员变量的实现,__MyObject__init_block_impl_0是代码里block的实现,各自的实现中已经没有了__strong修饰符了。
但是在MRC下,我们对Block执行了copy,它会调用__MyObject__init_block_copy_0方法,而该方法会调用_Block_object_assign方法,我们可以阅读该方法对源代码,相关部分摘录如下:

void _Block_object_assign(void *destAddr, const void *object, const int flags)
{
  // ......
    if (IS_SET(flags, BLOCK_FIELD_IS_OBJECT) &&
          !IS_SET(flags, BLOCK_BYREF_CALLER))
    {
        id src = (id)object;
        void **dst = destAddr;
        *dst = src;
        if (!isGCEnabled)
        {
            *dst = objc_retain(src);
        }
    }
}

所以在执行完copy之后,会导致self的引用计数加1,但是代码中却没有对应的释放引用计数的代码,所以会导致MyObject对象无法释放,同样Block也无法释放。

破解循环引用的原理

ARC下破解循环引用的原理

我们知道在ARC下破解循环引用的方法是使用__weak修饰符,如下代码:

__weak MyObject *tmp = self;
blk = ^{NSLog(@"self = %@", tmp);};

修改后,转换的主要代码如下:

因为使用了__weak,需要在转换的时候指定objc-runtime参数:-fobjc-runtime=ios-11.0.0

struct MyObject_IMPL {
    struct NSObject_IMPL NSObject_IVARS;
    __strong blk_t blk;
};

struct __MyObject__init_block_impl_0 {
  struct __block_impl impl;
  struct __MyObject__init_block_desc_0* Desc;
  MyObject *__weak tmp;
  __MyObject__init_block_impl_0(void *fp, struct __MyObject__init_block_desc_0 *desc, MyObject *__weak _tmp, int flags=0) : tmp(_tmp) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

可以看到,在block里对self的引用已经变成了__weak的了,这样就不会再有循环引用了。

MRC下破解循环引用的原理

在MRC环境下,我们不能使用__weak修饰符来解决这个问题了。也许聪明的看官已经知道了一种破解方法:既然是因为copy导致的引用计数加1,那么在Block体内做一次release就好了,如下:

blk = [^{
  NSLog(@"self = %@", self);
  [self release];
} copy];

经过实验,这种方法确实能解决循环引用的问题,但是这种编码方式不太符合苹果的内存管理原则(简单来说,retain和release是对应的),所以代码看起来会有点怪异。
解决MRC下循环引用的关键在于在对Block执行copy的时候不增加self的引用计数,我们有另外一种方案:

__block MyObject *tmp = self;
blk = [^{NSLog(@"self = %@", tmp);} copy];

经过转化后的主要代码如下:

struct MyObject_IMPL {
    struct NSObject_IMPL NSObject_IVARS;
    blk_t blk;
};

struct __Block_byref_tmp_0 {
  void *__isa;
__Block_byref_tmp_0 *__forwarding;
 int __flags;
 int __size;
 void (*__Block_byref_id_object_copy)(void*, void*);
 void (*__Block_byref_id_object_dispose)(void*);
 MyObject *tmp;
};

struct __MyObject__init_block_impl_0 {
  struct __block_impl impl;
  struct __MyObject__init_block_desc_0* Desc;
  __Block_byref_tmp_0 *tmp; // by ref
  __MyObject__init_block_impl_0(void *fp, struct __MyObject__init_block_desc_0 *desc, __Block_byref_tmp_0 *_tmp, int flags=0) : tmp(_tmp->__forwarding) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

static void __MyObject__init_block_copy_0(struct __MyObject__init_block_impl_0*dst, struct __MyObject__init_block_impl_0*src) {_Block_object_assign((void*)&dst->tmp, (void*)src->tmp, 8/*BLOCK_FIELD_IS_BYREF*/);}

对Block执行copy的时候会调用__MyObject__init_block_copy_0方法,该方法会调用_Block_object_assign方法,这个方法会做如下操作:

void _Block_object_assign(void *destAddr, const void *object, const int flags)
{
    if (IS_SET(flags, BLOCK_FIELD_IS_BYREF))
    {
        struct block_byref_obj *src = (struct block_byref_obj *)object;
        struct block_byref_obj **dst = destAddr;
        src = src->forwarding;

        if ((src->flags & BLOCK_REFCOUNT_MASK) == 0)
        {
            // ...
        }
        else
        {
            *dst = (struct block_byref_obj*)src;
            increment24(&(*dst)->flags);
        }
    }
}

由以上源代码可以得知,在copy的时候没有对self执行retain,所以不会增加self的引用技术。

总结

本文我们分别分析了MRC和ARC环境下循环引用产生的原因和解决原理,在MRC下一般推荐使用__block,在ARC下推荐使用__weak,对于self变量,可以使用如下的宏定义:

#ifndef    weakify
    #if __has_feature(objc_arc)
        #define weakify autoreleasepool{} __weak __typeof__(self) weakRef = self;
    #else
        #define weakify autoreleasepool{} __block __typeof__(self) blockRef = self;
    #endif
#endif

使用语法是@weakify,在ARC下会生成weakRef变量,在MRC下会生成blockRef变量。

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

推荐阅读更多精彩内容

  • 本系列博文总结自《Pro Multithreading and Memory Management for iOS...
    McDan阅读 608评论 0 1
  • 《Objective-C高级编程》这本书就讲了三个东西:自动引用计数、block、GCD,偏向于从原理上对这些内容...
    WeiHing阅读 9,780评论 10 69
  • Blocks Blocks Blocks 是带有局部变量的匿名函数 截取自动变量值 int main(){ ...
    南京小伙阅读 899评论 1 3
  • Block 梳理与疑问 时隔一年,再次读 《Objective-C 高级编程》,看到 block 一章,这一次从头...
    DeerRun阅读 619评论 0 2
  • block.png iOS代码块Block 概述 代码块Block是苹果在iOS4开始引入的对C语言的扩展,用来实...
    全栈农民工阅读 588评论 0 1