iOS-Block

Block是一种匿名函数,也是一种Objective-C对象。

语法


^ 返回值 (参数列表) 表达式
^ int (int a) {return a+1}

返回值和参数列表都可以省略

^ 表达式
^{NSLog(@"abc")}

声明


block可以作为一个临时变量,也可作为方法上的参数,更可以作为一个函数定义。

// 临时变量 - 
// 返回值类型 (^变量名) (参数列表) = 语法
int (^aBlock)(int) = ^ int (int a) {return a+1};
NSLog(@"%d", aBlock(10)); 

// 方法参数 -
(返回值 (^) (参数列表)) 参数名
- (void)excute:(void (^) (int a))blk;

// 函数 -
typedef 返回值类型 (^函数名) (参数列表)
typedef int (^MyBlock)(int a);
MyBlock blk;
- (void)excute:(MyBlock)blk;

作为变量和作为方法参数Block的格式有一点区别,为了避免记忆上的麻烦,建议使用函数形式定义。

局部变量


函数中的局部变量在block中使用时,其值不受后续代码影响。

int a = 10;
void (^MyBlock)() = ^{
    NSLog(@"内部 a=%d", a); // 输出 10
    a = 30; // 编译器报错
};
a = 20;
MyBlock();
NSLog(@"外部 a=%d", a); // 输出20

因为block在编译时会转化为普通的C语言代码,使用了struct结构。在block中使用变量a时,其实质是在block中声明了相同属性的变量,姑且称之为copyA,并且将a的值赋予了copyA。在block使用的实际上是copyA。a的值在声明了block之后才发生改变,按代码的编译顺序并不会影响copyA,因此输出的值还是10。

__block


通常情况下block中不允许对外部局部变量重新赋值,除非该变量是静态局部变量、静态全局变量或者成员变量。如果需要改变局部变量的值,可以采用__block修饰符进行修饰。

__block int a = 10;
void (^MyBlock)() = ^{
    NSLog(@"内部 a=%d", a); // 输出 20
    a = 30;
};
a = 20;
MyBlock();
NSLog(@"外部 a=%d", a); // 输出30

究其原因,使用__block修饰符修饰的变量在编译时,block会为其创建一个结构体,结构体中保留了该变量的地址。在使用该变量时,实际使用的是指针的方式访问,因此不管是在函数中或者是在block中改变该变量都会互相影响。

strong / copy


当block被作为一个成员变量时,该使用strong还是copy呢?
在block转换成结构体实例时会使用到objc_retainBlock函数,而该方法在runtime时实际上就是_Block_copy函数,因此使用copy即可。
使用copy方法会将block从栈上复制到堆上,因此当栈上的block被废弃时(超出作用域)还能继续使用该block。

循环引用


在block中使用某个对象时,block会持有该对象,在内部形成一个类似autorelease的对象。当block作为局部变量时,在内部使用了self并不会引起循环引用。但当block作为成员变量时,由于self持有了该block,而block又持有了self,就会导致循环引用,无法释放内存。因此如果需要在block中使用某个对象,通常建议使用该对象的弱引用。

BLK blk = ^{
    NSLog(@"self=%@", self); // 不会造成循环引用
};
blk();
    
self.blk2 = ^{
    NSLog(@"self=%@", self); // 循环引用,编译器警告
};
self.blk2();

__weak ViewController * weakSelf = self;
self.blk3 = ^{
    NSLog(@"self=%@", weakSelf); // 不会引起循环引用
};
self.blk3();
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 前言 Blocks是C语言的扩充功能,而Apple 在OS X Snow Leopard 和 iOS 4中引入了这...
    小人不才阅读 3,785评论 0 23
  • 《Objective-C高级编程》这本书就讲了三个东西:自动引用计数、block、GCD,偏向于从原理上对这些内容...
    WeiHing阅读 9,900评论 10 69
  • Block是iOS开发中一种比较特殊的数据结构,它可以保存一段代码,在合适的地方再调用,具有语法简介、回调方便、编...
    飞鱼湾阅读 4,164评论 0 7
  • 《Objective-C高级编程》是一本有趣又难懂的书,全书就讲了引用计数、Block、GCD三个概念,有趣是因为...
    kamous阅读 55,365评论 22 206
  • 工具命令转化C++xcrun -sdk iphoneos clang -arch arm64 -rewrite-o...
    iYeso阅读 3,149评论 6 67