你想知道Objective-C中blocks是怎么工作的吗?接下来就让我们通过几个测试题来了解下吧。
1.ARC转MRC MRC转ARC
首先, 先要了解下ARC转MRC和MRC转ARC.
ARC项目转MRC
创建项目, 打开Xcode, 点击项目, 找到Build Phases中的Compile Sources, 将需要转为MRC的 .m文件加入编译标记 -fno-objc-arc
MRC项目转ARC
给MRC项目中.m文件添加ARC标记 -fobjc-arc
2.示例演示
Example A
void exampleA() {
char a = 'A';
^{
printf("%c\n", a);
}();
}
1.ARC和MRC都有效 ?
2.只有ARC 有效?
3.MRC有效 ?
4.都没有效?
Example B
void exampleB_addBlockToArray(NSMutableArray *array) {
char b = 'B';
[array addObject:^{
printf("%c\n", b);
}];
}
void exampleB() {
NSMutableArray *array = [NSMutableArray array];
exampleB_addBlockToArray(array);
void (^block)(void) = [array objectAtIndex:0];
block(); //MRC 断点查看下 __NSStackBlock__
NSLog(@"%@", [block class]);// ARC __NSMallocBlock__
}
1.ARC和MRC都有效 ?
2.只有ARC 有效?
3.MRC有效 ?
4.都没有效?
Example C
void exampleC_addBlockToArray(NSMutableArray *array) {
[array addObject:^{
printf("C\n");
}];
}
void exampleC() {
NSMutableArray *array = [NSMutableArray array];
exampleC_addBlockToArray(array);
void (^block)(void) = [array objectAtIndex:0];
block();
NSLog(@"%@", [block class]); // __NSGlobalBlock__
}
1.ARC和MRC都有效 ?
2.只有ARC 有效?
3.MRC有效 ?
4.都没有效?
Example D
typedef void (^dBlock)(void);
dBlock exampleD_getBlock() {
char d = 'D';
return ^{
printf("%c\n", d);
} ;
}
void exampleD() {
exampleD_getBlock()();
}
1.ARC和MRC都有效 ?
2.只有ARC 有效?
3.MRC有效 ?
4.都没有效?
Example E
typedef void (^eBlock)(void);
eBlock exampleE_getBlock() {
char e = 'E';
void (^block)(void) = ^{
printf("%c\n", e);
};
return block;
}
void exampleE() {
eBlock block = exampleE_getBlock();
block(); //MRC 断点查看下 __NSStackBlock__
NSLog(@"%@", [block class]); // ARC __NSMallocBlock__
}
1.ARC和MRC都有效 ?
2.只有ARC 有效?
3.MRC有效 ?
4.都没有效?
3.解析
Example A: ARC和MRC都有效
不管在 ARC 还是 MRC 下,不论 block 存放在 stack 还是 heap 内存中,当example A 被调用时,block 仍然有效,都能正常执行.
Example B: 只有ARC 有效
在 MRC 下,exampleB_addBlockToArray 中的 block 是 NSStackBlock 类型,存放在stack内存中。当执行 exampleB 时,stack 内存被释放,block 失效.
在 ARC 下,block 是 autoreleased NSMallocBlock 类型,存放在 heap 内存中,所以 Exmaple B 只有ARC 有效.
Example C: ARC和MRC都有效
当 block 不需要从外部获取变量时,它不需要在 runtime 设置任何状态。此时,block 被编译成 NSGlobalBlock 类型,放在内存 data 段,就像 C 函数一样,属于代码的一部分,所以 ARC和MRC都有效.
Example D: 只有ARC 有效
这题有点类似于 Example B. 在 MRC 下,exampleD_getBlock 中的block 会被创建在 stack 内存中,当函数返回时,block马上失效。鉴于本题的错误实在太明显,编译器在编译时,就会抛出错误 error: returning block that lives on the local stack.
而在 ARC 下,block 会被编译成 autoreleased NSMallocBlock 类型,存放于 heap 内存中。
所以 只有ARC 有效.
Example E: 只有ARC 有效
本题类似于 Example D,区别在于本题代码不会出现编译错误,而是在运行时才会崩溃。更槽糕的是,如果你关闭了编译器优化选项,代码运行正常,而无法发现这个隐藏的bug。
而在 ARC 下,block 会被编译成 autoreleased NSMallocBlock 类型,存放于 heap 内存中。
所以 只有ARC 有效.
4.总结
总结1:
MRC 中block 没有引用外部变量, block为 NSGlobalBlock 类型,存储在全局数据区.
MRC 中block 引用外部变量,block为NSStackBlock 类型,存储在栈内存中.
所以, 在block所属的栈作用域外使用block时, 需要将调用copy方法将该block存储在堆区.
总结2:
ARC 中 没有引用外部变量, block为 NSGlobalBlock 类型,存储在全局数据区.
ARC 中 引用外部变量, block为 autoreleased NSMallocBlock 类型,存储在堆内存中.
例外:匿名block 引用外部变量
在ARC下其实很少见到 NSStackBlock 类的Block,大多数情况编译器都保证了Block是在堆上创建的
int count = 11;
NSLog(@"Stack Block:%@", [^{NSLog(@"Stack Block:%d",count);} class]);
//打印:Stack Block:__NSStackBlock__
5.以上这么多例子告诉我们什么?
告诉我们要使用ARC!在ARC下,block总能正确运行。如果你不用ARC,最好能保证在 stack 内存中声明定义的block,能够拷贝到heap内存,保证block的正常运行。
由于笔者水平有限,文中如果有错误的地方,或者有更好的方法,还望大神指出。
附上本文的所有 demo 下载链接,【GitHub】。
如果你看完后觉得对你有所帮助,还望在 GitHub 上点个 star。赠人玫瑰,手有余香。
网友评论