栈块、堆块、全局块

编译器版本为Clang5,主要技术是ARC,参考来自Objective-C Automatic Reference Counting (ARC)

block根据分配的内存位置分为栈块,堆块,全局块。

一个block在创建的时候,就可以确定其类型,判断一个block是那种类型其实很简单,规则如下(MRC和ARC都适用):

1.如果一个block中引用了全局变量,或者没有引用任何外部变量(属性、实例变量、局部变量),那么该block为全局块。
2.其它引用情况(局部变量,实例变量,属性)为栈块。

上面的规则中并不存在堆块的判断,因为以上规则的前提条件是:block在创建的时候,不是创建完之后赋值给变量,代码如下:

@interface MGBlockExample()
//属性
@property(nonatomic,assign)int a;

@end
@implementation MGBlockExample{
    int _b; //实例变量
}
//全局变量
int c = 3;

- (void)test{
    
    /*****      全局块        *****/
    
    //引用全局变量
    NSLog(@"aBlock:%@", ^{ NSLog(@"c:%d",c); }); //aBlock:__NSGlobalBlock__
    
    //没有引用变量
    NSLog(@"bBlock:%@", ^{ NSLog(@""); }); //bBlock:__NSGlobalBlock__
    
    
    /*****       栈块        *****/
    
    int d = 4;
    //不给赋值,但引用了局部变量
    NSLog(@"cBlock:%@",^void{ NSLog(@"d:%d",d); }); //cBlock:__NSStackBlock__
    //不给赋值,但引用了实例变量
    NSLog(@"dBlock:%@",^void{ NSLog(@"_b:%d",_b); }); //dBlock:__NSStackBlock__
    //不给赋值,但引用了属性
    NSLog(@"eBlock:%@",^void{ NSLog(@"self.a:%d",self.a); }); //eBlock:__NSStackBlock__
}
@end

全局块和栈块已经有规则可以判断,并且适用于MRC和ARC,规则的前提条件是block创建的时候。那么在将创建完的块之后赋值给变量会出现什么情况呢?看下面代码:

- (void)test1{
    //局部变量
    void (^fBlock)() = ^(){
        NSLog(@"d:%d",d);
    };
    
    //实例变量
    void (^gBlock)() = ^(){
        NSLog(@"_b:%d",_b);
    };
    
    //属性
    void (^hBlock)() = ^(){
        NSLog(@"self.a:%d",self.a);
    };
    NSLog(@"fBlock:%@", fBlock); 
    NSLog(@"gBlock:%@", gBlock); 
    NSLog(@"hBlock:%@", hBlock); 
    //MRC:上面操作输出//XBlock:__NSStackBlock__
    //ARC:上面操作输出//XBlock: __NSMallocBlock__
}

从上面代码可以看到在MRC环境中,block引用局部变量、实例变量、属性的的结果依然是生成栈块。而在ARC环境中,生成的是堆块。原因如下:

block是retainable object pointer类型。

在ARC环境中,ARC会对retainable type进行内存管理:

If an object is declared with retainable object owner type, but without an explicit ownership qualifier, its type is implicitly adjusted to have __strong qualification.
如果一个被定义对象是retainable object owner类型,并且没有明确指定内存管理策略,那么编译器会默认为其添加__strong修饰词。

所以上面的fBlock在ARC环境中是这样的:

__strong void (^fBlock)() = ^(){
        NSLog(@"d:%d",d);
    };

然后document又说:

With the exception of retains done as part of initializing a __strong parameter variable or reading a __weak variable, whenever these semantics call for retaining a value of block-pointer type, it has the effect of a Block_copy. The optimizer may remove such copies when it sees that the result is used only as an argument to a call.
对于block来说,不管是在创建对象的时候strong修饰还是在读取weak修饰的变量,其效果和用Block_copy修饰是一样的。当编译器觉得可以优化的时候就不会执行Block_copy函数,例如一个block初始化完之后就被当做一个参数使用了(例如cBlock、dBlock、eBlock)

所以上面fBlock的代码转成MRC是类似这样的:

void (^fBlock)() = [^(){
        NSLog(@"d:%d",d);
    } copy];

给block对象的-copy方法发送消息,会将该对象拷贝一份放入堆中,所以这个时候块从栈块变成堆块。


补充一下block的内存管理

不管是对block进行retiancopyrelease,block的引用计数都不会增加,始终为1。

NSGlobalBlock:使用retaincopyrelease都无效,block依旧存在全局区,且没有释放, 使用copyretian只是返回block的指针。

NSStackBlock:使用retainrelease操作无效;栈区block会在方法返回后将block空间回收; 使用copy将栈区block复制到堆区,可以长久保留block的空间,以供后面的程序使用。

NSMallocBlock:支持retianrelease,虽然block的引用计数始终为1,但内存中还是会对引用进行管理,使用retain引用+1, release引用-1; 对于NSMallocBlock使用copy之后不会产生新的block,只是增加了一次引用,类似于使用retian

总结起来是这样的:

1.在创建block的时候(MRC和ARC通用):

1.1. 如果一个block中引用了全局变量,或者没有引用任何外部变量(属性、实例变量、局部变量),那么该block为全局块。
1.2. 其它引用情况(局部变量,实例变量,属性)为栈块。

2.在将block对象赋值给其它对象^(oBlock)()的时候(ARC):

2.1. 如果block是栈块,那么oBlock变成堆块(因为Clang编译器帮我们往block发送了copy消息)。
2.2. 如果block是全局块,那么oBlock也是全局块,如果block是堆块,那么oBlock也是堆块。
2.3. 如果仅把创建完的 block 当做方法参数使用(不进行其它赋值引用),那么 block 的类型不变(参照1.1和1.2的情况)。

3.在将block对象赋值给其它对象^(oBlock)()的时候(MRC):

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

推荐阅读更多精彩内容

  • 内存管理 简述OC中内存管理机制。与retain配对使用的方法是dealloc还是release,为什么?需要与a...
    丶逐渐阅读 1,964评论 1 16
  • 《Objective-C高级编程》这本书就讲了三个东西:自动引用计数、block、GCD,偏向于从原理上对这些内容...
    WeiHing阅读 9,810评论 10 69
  • iOS代码块Block 概述 代码块Block是苹果在iOS4开始引入的对C语言的扩展,用来实现匿名函数的特性,B...
    smile刺客阅读 2,347评论 2 26
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,140评论 30 470
  • 为什么进行内存管理? 由于移动设备的内存极其有限,所以每个APP所占的内存也是有限制的,当app所占用的内存较多时...
    天天想念阅读 894评论 1 7