block
Block简介
block�实际上是Objective-c对闭包的实现。闭包允许一个函数访问声明该函数运行上下文中的变量,甚至可以访问不同运行上下文中的变量。
Block使用场景
通常作为回调函数取代传统的回调方式。主要使用场景为:
- 任务完成时回调处理。
- 消息监听时的回调处理。
- 错误消息的回调处理。
- 枚举回调。
- 视图动画、变换。
- 排序。
Block的结构
返回值类型(^Block名)(形参类型 形参,...)= ^(形参类型 形参,...){.......};
调用执行:Block名(形参类型 形参,...);
Block的使用
Block的使用类比与函数指针。
声明→赋值→调用
void(^block)(int a, int b); block = ^(int a, int b){ a = a+b; NSLog(@"++++++%d", a); }; block(3, 4);
定义Block类型方式:
typedef void(^BlockType)(int a, int b); BlockType tempBlock = ^(int a, int b){ a = a+b; NSLog(@"++++++%d", a); }; tempBlock(5, 6);
作为函数参数方式:
- (void)useBlockIdentifier:(void(^)(int a, int b))blockIdentifier{ blockIdentifier(7, 8); }
[self useBlockIdentifier:tempBlock];
修改闭包变量的值:
默认通过Block进行闭包访问的变量是const的,如果对其修改会报错
int a = 5; void (^testBlock)() = ^{ a++; NSLog(@"------%d",a); }; testBlock();
编译器会报错:Variable is not assignable (missing __block type specifier)
原因:Block默认是将�变量复制到其数据结构中来实现访问的。
加上__block即可实现对变量的修改,原因:__block修饰的外部变量引用,block是复制其引用地址来实现访问的。
Block对象的生存周期
MRC模式下
NSGlobalBlock,无引用外部变量,位于代码段。
typedef void(^BlockType)(int a, int b); BlockType tempBlock = ^(int a, int b){ a = a+b; NSLog(@"++++++%d", a); }; NSLog(@"tempBlock::%@", tempBlock)
输出:tempBlock::<__NSGlobalBlock__: 0x105539183>
NSStackBlock,有引用外部变量,位于栈上。
__block int a = 10; void (^testBlock)() = ^{ a++; NSLog(@"------%d",a); }; NSLog(@"testBlock::%@", testBlock);
输出:testBlock::<__NSStackBlock__: 0x7fff54e43200>
NSMallocBlock,对于NSStackBlock类型进行copy/Block_copy()得到,位于堆上。
typedef void(^BlockType)(int a, int b); BlockType testAllocBlock = [testBlock copy]; NSLog(@"testAllocBlock::%@", testAllocBlock);
输出:testAllocBlock::<__NSMallocBlock__: 0x7fef3f4251d1>
小结
* | retain | copy/Block_copy() | release/Block_release() |
---|---|---|---|
NSGlobalBlock | 无用 | 无用 | 无用 |
NSStackBlock | 无用 | 从栈上拷贝一份到堆上,拷贝后的retainCount + 1 | 无用 |
NSMallocBlock | retainCount + 1 | retainCount + 1,不创建新的对象 | retainCount - 1,当retainCount为0时,block无效,指向其的指针变成野指针 |
对于函数返回block应使用copy+autorelease。
return [[stackBlock copy] autorelease];
对于块属性应使用copy关键字。
@property (copy, nonatomic) void (^myBlock)(int , int);
ARC
NSGlobalBlock,无引用外部变量,位于代码段。
typedef void(^BlockType)(int a, int b); BlockType tempBlock = ^(int a, int b){ a = a+b; NSLog(@"++++++%d", a); }; NSLog(@"tempBlock::%@", tempBlock)
输出:tempBlock::<__NSGlobalBlock__: 0x105536540>
NSStackBlock,有引用外部变量,且用weak修饰,位于栈上。
__block int a = 5; __weak void (^testBlock)() = ^{ a++; NSLog(@"------%d",a); }; NSLog(@"testBlock::%@", testBlock);
输出:testBlock::<__NSStackBlock__: 0x7fff54231e00>
NSMallocBlock,有引用外部变量,用strong修饰或者无修饰,位于堆上。
用strong修饰:
__strong void (^testBlock)() = ^{ a++; NSLog(@"------%d",a); }; NSLog(@"testBlock::%@", testBlock);
输出:testBlock::<__NSMallocBlock__: 0x7feb2ac2e320>
无修饰:
void (^testBlock)() = ^{ a++; NSLog(@"------%d",a); }; NSLog(@"testBlock::%@", testBlock);
输出:testBlock::<__NSMallocBlock__: 0x7ff9c43215e0>