浅谈Block

前言

block的具体怎么使用我在这里就不一一细说, 我主要说的是关于block三种类型之间的区别, 以及block的内存管理

Block简介

Block字面意思就是代码块iOS4.0Mac OS X 10.6开始在Apple引入的特性
BlockObjective C语言中的对象 但是与NSObject有所区别 Block是特殊的Objective C对象

iOS中内存分区可以分为5个区:
block默认建立在栈区,如果block离开了方法作用域,block所占用的空间就会被回收掉
注: 在ARC下,系统在大部分情况下,会将block从栈上复制到堆上,这个后面会细说

说到内存分区,内存即指的是RAM。

  • 栈区(stack): 这个一般由编译器操作,或者说是编译器自动分配释放 ,会存一些局部变量,函数跳转跳转时现场保护(寄存器值保存于恢复),这些系统都会帮我们自动实现,无需我们干预。 所以大量的局部变量,深递归,函数循环调用都可能耗尽栈内存而造成程序崩溃 。
  • 堆区(heap): 一般由程序员管理,比如alloc申请内存,free释放内存。我们创建的对象也都放在这里。
  • 全局区(静态区 static):全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另一块区域。程序结束后有系统释放。注意:在嵌入式系统中全局区又可分为未初始化全局区:.bss段和初始化全局区:data段。举例:int a;未初始化的。int a = 10;已初始化的。
  • 常量区:常量字符串就是放在这里的,还有const常量。程序结束释放 NSString *lastName = @“xue”;
  • 代码区:存放代码,app程序会拷贝到这里

block有三种类型:

NSGlobalBlock NSStackBlock NSMallocBlock

1. NSGlobalBlock

看名字可知该block是存储在内存中全局区的
无论是在ARC还是MRC中 这个block在控制台输出的都是NSGlobalBlock

  void(^globalBlock)(void) = ^{

    };
    NSLog(@"------>%@",globalBlock);

控制台输出:<__NSGlobalBlock__: 0x10ba430e0>

结论: 只要实现一个对周围变量没有引用的block,就会显示为是NSGlobalBlock

切换到MAC只需要选中 TARGETS ->Build Setting之后再输入栏中输入automatic 找到Objective-C Automatic Reference Counting YES改成NO即可

2. NSStackBlock

这里先说一下block捕获外部变量

block内可以访问block之前定义的变量:但是不能修改

    NSInteger a = 10;
    void(^globalBlock)(void) = ^{
        NSLog(@"----->%zd",a);
    };

但是,如果想在block内部改变a的值,加上__block修饰符即可 用__block修饰之后,系统会传递a的地址(&a)

    __block NSInteger a = 10;
    void(^globalBlock)(void) = ^{
        a ++;
        NSLog(@"----->%zd",a);
    };

如果变量astaticstatic global或者global变量,则不需要添加__block,该值也是可以在block内部修改的。

因为staticstatic global或者global变量都是存储在内存中的全局区(静态区),对于这三种类型变量,block内部是捕获了其指针,则可以直接访问修改;而对于之前的临时变量,block则只是捕获了该变量的值,无法修改到外部的变量。

MRC环境下

    __block NSInteger a = 10;
    void(^globalBlock)(void) = ^{
        a ++;
    };
    NSLog(@"------>%@",globalBlock);
    NSLog(@"------>%@",[globalBlock copy]);

控制台打印:
<__NSStackBlock__: 0x7fff58ba0960>
<__NSMallocBlock__: 0x600000241ef0>

ARC环境下

控制台打印:
<__NSMallocBlock__: 0x60800025a730>
<__NSMallocBlock__: 0x60800025a730>

结论:

  1. MRC下只要引用了变量就是NSStackBlock类型, NSStackBlock类型copy之后就是__NSMallocBlock__
  2. ARC下,系统在大部分情况下,会将block从栈上复制到堆上,后面会细说

3. NSMallocBlock

MRC__NSStackBlock__ copy之后就是NSMallocBlock
ARC大部分情况下, 系统会默认的把block从栈上复制到堆上

ARC 下 block 的自动拷贝和手动拷贝

ARCblock的自动拷贝和手动拷贝

1.作为方法返回值
2.将Block赋值给附有__strong修饰符的id类型的类或者Blcok类型成员变量时
3.在方法名中含有usingBlockCocoa框架方法或者GDC的API中传递的时候.(比如使用NSArrayenumerateObjectsUsingBlockGCDdispatch_async方法时,其block不需要我们手动执行copy操作)系统方法内部对block进行了copy操作

因为在ARC下,对象默认是用__strong修饰的,所以大部分情况下编译器都会将block从栈自动复制到堆上,除了以下情况

block 作为方法或函数的参数传递时,编译器不会自动调用 copy 方法;
block 作为临时变量,没有赋值给其他block

block中对象的内存管理

以下都是在MAC环境下
新建一个Father类 里面声明一个属性name

声明代码:
{
    Father  * _globalFather;//全局变量
    
}
@property (nonatomic, copy) myBlock  block;
@property (nonatomic, strong) Father    * instanceFather;//实例变量

   Father * localFather = [[Father alloc] init];
   _globalFather = [[Father alloc] init];
   self.instanceFather = [[Father alloc] init];
   void(^retainCountBlock)() = ^(){
       localFather.name = @"li";
       _globalFather.name = @"li";
      self.instanceFather.name = @"li";
   };
   NSLog(@"%zd---%zd---%zd---%zd",localFather.retainCount,_globalFather.retainCount,self.instanceFather.retainCount,self.retainCount);
   [retainCountBlock copy];
   NSLog(@"%zd---%zd---%zd---%zd",localFather.retainCount,_globalFather.retainCount,self.instanceFather.retainCount,self.retainCount);
打印结果:
一:
  void(^retainCountBlock)() = ^(){
       localFather.name = @"li";
   };
控制台输出: 
1---1---2---5
2---1---2---5

二:
  void(^retainCountBlock)() = ^(){
       _globalFather.name = @"li";
   };
控制台输出: 
1---1---2---5
1---1---2---6

三:
 void(^retainCountBlock)() = ^(){
      self.instanceFather.name = @"li";
   };
控制台输出: 
1---1---2---5
1---1---2---6

结论:
1. localFatherblock Copy时,系统自动增加引用计数
2._globalFather是当前类的属性,在blockCopy时, _globalFather引用计数没增加,造成self引用计数增加
3. 如果在block中使用了self也会增加self的引用计数
2和3都会造成self引用计数增加,造成循环引用, 解决循环引用只需要 加上 __weak修饰一下即可

最后总结

Block是默认建立在栈上, 所以如果离开方法作用域, Block就会被丢弃
ARC下 : 对象默认是用__strong修饰的,所以大部分情况下编译器都会将block从栈自动复制到堆上
MRC下 : 只要实现一个对周围变量没有引用的Block,就会显示为是NSGlobalBlock
如果其中加入了变量的引用,就是NSStackBlock
如果你对一个NSStackBlock对象使用了Block_copy()或者发送了copy消息,就会得到NSMallocBlock

  • NSGlobalBlockretaincopyrelease操作都无效;
  • NSStackBlockretainrelease操作无效,必须注意的是,NSStackBlock在函数返回后,Block内存将被回收。即使retain也没用。容易犯的错误是[mutableAarry addObject:stackBlock],(补:在ARC中不用担心此问题,因为ARC中会默认将实例化的Block拷贝到堆上)在函数出栈后,从mutableAarry中取到的stackBlock已经被回收,变成了野指针。正确的做法是先将[stackBlock copy]到堆上,然后加入数组:[mutableAarry addObject:[[stackBlock copy] autorelease]]。支持copycopy之后生成新的NSMallocBlock类型对象。
  • NSMallocBlock支持retainrelease,虽然retainCount始终是1,但内存管理器中仍然会增加、减少计数。copy之后不会生成新的对象,只是增加了一次引用,类似retain
  • Block_copycopy等效,Block_releaserelease等效;
  • Block不管是retaincopyrelease都不会改变引用计数retainCountretainCount始终是1;

参考文献: 学会使用Objective-C中的block

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

推荐阅读更多精彩内容

  • 前言 Blocks是C语言的扩充功能,而Apple 在OS X Snow Leopard 和 iOS 4中引入了这...
    小人不才阅读 3,753评论 0 23
  • 1. iOS Block用来封装一段代码块或者传递参数相对于代理使用起来方便,它本质上是一个匿名函数。 2.使用b...
    extanstory阅读 321评论 0 0
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,121评论 29 470
  • 概述闭包,可以捕获代码。block对象常量块外的属性 非标准c组成部分。源码部分不粘了 - 主要证明两点 1 是个...
    zx1798阅读 725评论 1 2
  • 《Objective-C高级编程》这本书就讲了三个东西:自动引用计数、block、GCD,偏向于从原理上对这些内容...
    WeiHing阅读 9,792评论 10 69