ARC环境下Block的内存管理

对于Block的相关知识,可以看《Objective-C高级编程 iOS与OS X多线程和内存管理》这本书,写得非常透彻。

一、Block是什么?

Block是C语言的扩充功能。是带有自动变量(局部变量)的匿名函数。

Block 也是 Objective-C 对象,将Block 当作 Objective-C 对象来看时,该 Block 的类为 _NSConcreteStackBlock

二、Block有几种类型?

3种:

 _NSConcreteStackBlock      该类的对象Block设置在栈上
 _NSConcreteGlobalBlock     与全局变量一样,设置在程序的数据区域(.data区)中
 _NSConcreteMallocBlock     该类的对象设置在由malloc函数分配的内存块(即堆)中

三、Block主要应用场景:

1.对象的属性;
2.方法的参数;
3.方法的返回值;

四、Block内存管理:

ARC环境,大多数情况下编译器会适当地进行判断,会自动生成将Block从栈上复制到堆上的代码。
Block作为函数返回值返回时,编译器会自动生成复制到堆上的代码。但是有些情况需要我们手动生成代码将Block从栈上复制到堆上,使用“copy实例方法”。
编译器不能判断“自动将Block从栈上复制到堆上”的情况:向方法或函数的参数传递Block。但是如果方法或函数内部适当地复制了传递过来的参数,就不必在调用该方法或函数前手动复制了。例如,系统框架的含BlockAPI可不用手动调用copy方法复制。

以下方法中 Block 作为参数,必须调用 copy 方法,否则会导致程序异常退出。

-(id)getBlockArray
{
    int val = 10;
    //Block变量类型可以直接调用copy方法。所以说Block其实也是Objective-C对象。
    //不管Block配置在堆、栈或者数据区域,用copy方法复制都不会引起任何问题。
    return [[NSArray alloc] initWithObjects:[^{NSLog(@"blk0:%@",@(val));} copy],[^{NSLog(@"blk1:%@",@(val));} copy], nil];
}

- (void)viewDidLoad {
    [super viewDidLoad];

    //正常执行。
    id obj = [self getBlockArray];
    blk_t blk = (blk_t)[obj objectAtIndex:0];
    blk();
}

Demo地址:https://github.com/xiaoL0204/StackBlockDemo

如果不调用copy方法,会报如下错误。这通常是由野指针引起的,说明Block对象被释放了。

图1.不使用copy方法会Crash

通过Block的复制,__block变量也从栈复制到堆。此时可同时访问栈上的__block变量和堆上和__block变量。

五、什么时候栈上的Block会复制到堆呢?

1.调用Blockcopy实例方法时;
2.Block作为函数返回值返回时;
3.将Block赋值给附有__strong修饰符id类型的类或Block类型成员变量时;
4.在方法名中含有usingBlockCocoa框架方法或Grand Central DispatchAPI中传递Block时。

这些情况下,可归结为_Block_copy函数被调用时Block从栈复制到堆。释放Block时调用其dispose函数,相当于对象的dealloc实例方法。

六、Block特性:截获对象,对象可超出其变量作用域而存在:

Block中使用的赋值给附有__strong修饰符的自动变量的对象和复制到堆上的__block变量由于被堆上的Block所持有,因而可超出其变量作用域而存在。

如果不调用Blockcopy实例方法,Block不会调用_Block_copy函数,即使截获了对象,它也会随着变量作用域的结束而被释放。

超出作用域截获对象示例:

//Block截获对象,对象超出变量的作用域而存在。
    id array = [NSMutableArray array];
    void (^blk_t2) (id obj) = [^(id obj){
        [array addObject:obj];
        
        NSLog(@"array count = %@,obj:%@",@([array count]),obj);
        
    } copy];
    
    
    blk_t2([[NSObject alloc] init]);
    blk_t2([[NSObject alloc] init]);
    blk_t2([[NSObject alloc] init]);
    
    NSLog(@"array:%@",array);

执行结果:

图2.超出作用域截获对象

七、Block循环引用

如果在Block中使用__strong修饰符的对象类型自动变量,那么当Block从栈复制到堆时,该对象为Block持有。当对象持有Block,且该Block持有该对象时,会引起循环引用。
Block内部没有显示调用self也可能引起循环引用。

图3.Block循环引用
图4.破坏Block循环引用

使用__weak修饰符修饰会相互持有的变量,在Block内部使用该变量即可避免循环引用。
NSTimer在作为控制器属性的时候容易产生循环引用,这点跟Block循环引用很类似。因为NSTimer会持有target对象,除非NSTimer被置为nilinvalidate或停止,否则timer会一直持有target对象,如果此时target对象持有这个timer对象,就会循环引用,从而造成内存泄露。

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

推荐阅读更多精彩内容