本文来源于 ParseBlog 的其中一篇博文 《Objective-C Blocks Quiz》

如果您觉得我的博客对您有帮助,请通过关注我的新浪微博  MicroCai 支持我,谢谢!


你想知道Objective-C中blocks是怎么工作的吗?那么让我们通过几个测试题来了解下吧。 
本文所有的例子都经过以下版本的LLVM检验过:

 
  1. Apple clang version 4.1 (tags/Apple/clang-421.11.66) (based on LLVM 3.1svn)
  2. Target: x86_64-apple-darwin11.4.2
  3. Thread model: posix
 

Example A

 
  1. void exampleA() {
  2. char a = 'A';
  3. ^{
  4. printf("%c\n", a);
  5. }();
  6. }
  1. always works ?
  2. only works with ARC ?
  3. only works without ARC ?
  4. never works ?
 

Example B

 
  1. void exampleB_addBlockToArray(NSMutableArray *array) {
  2. char b = 'B';
  3. [array addObject:^{
  4. printf("%c\n", b);
  5. }];
  6. }
  7. void exampleB() {
  8. NSMutableArray *array = [NSMutableArray array];
  9. exampleB_addBlockToArray(array);
  10. void (^block)() = [array objectAtIndex:0];
  11. block();
  12. }
  1. always works ?
  2. only works with ARC ?
  3. only works without ARC ?
  4. never works ?
 

Example C

 
  1. void exampleC_addBlockToArray(NSMutableArray *array) {
  2. [array addObject:^{
  3. printf("C\n");
  4. }];
  5. }
  6. void exampleC() {
  7. NSMutableArray *array = [NSMutableArray array];
  8. exampleC_addBlockToArray(array);
  9. void (^block)() = [array objectAtIndex:0];
  10. block();
  11. }
  1. always works ?
  2. only works with ARC ?
  3. only works without ARC ?
  4. never works ?
 

Example D

 
  1. typedef void (^dBlock)();
  2. dBlock exampleD_getBlock() {
  3. char d = 'D';
  4. return ^{
  5. printf("%c\n", d);
  6. };
  7. }
  8. void exampleD() {
  9. exampleD_getBlock()();
  10. }
  1. always works ?
  2. only works with ARC ?
  3. only works without ARC ?
  4. never works ?
 

Example E

 
  1. typedef void (^eBlock)();
  2. eBlock exampleE_getBlock() {
  3. char e = 'E';
  4. void (^block)() = ^{
  5. printf("%c\n", e);
  6. };
  7. return block;
  8. }
  9. void exampleE() {
  10. eBlock block = exampleE_getBlock();
  11. block();
  12. }
  1. always works ?
  2. only works with ARC ?
  3. only works without ARC ?
  4. never works ?

 

解析

Example A: always works

不管在 ARC 还是 MRC 下,不论 block 存放在 stack 还是 heap 内存中,当example A 被调用时,block 仍然有效,都能正常执行.

Example B: only works with ARC

在 MRC 下,exampleB_addBlockToArray 中的 block 是 NSStackBlock 类型,存放在stack内存中。当执行 exampleB 时,stack 内存被释放,block 失效.

在 ARC 下,block 是 autoreleased NSMallocBlock 类型,存放在 heap 内存中,所以 Exmaple B only works with ARC.

Example C: always works

当 block 不需要从外部获取变量时,它不需要在 runtime 设置任何状态。此时,block 被编译成 NSGlobalBlock 类型,放在内存 data 段,就像 C 函数一样,属于代码的一部分,所以 always works.

Example D: only works with ARC

这题有点类似于 Example B. 在 MRC 下,exampleD_getBlock 中的block 会被创建在 stack 内存中,当函数返回时,block马上失效。鉴于本题的错误实在太明显,编译器在编译时,就会抛出错误 error: returning block that lives on the local stack.

而在 ARC 下,block 会被编译成 autoreleased NSMallocBlock 类型,存放于 heap 内存中。

所以 only works with ARC.

Example E: only works with ARC

本题类似于 Example D,区别在于本题代码不会出现编译错误,而是在运行时才会崩溃。更槽糕的是,如果你关闭了编译器优化选项,代码运行正常,而无法发现这个隐藏的bug。

而在 ARC 下,block 会被编译成 autoreleased NSMallocBlock 类型,存放于 heap 内存中。

所以 only works with ARC.


 

总结

以上这么多例子告诉我们什么?告诉我们要使用ARC!在ARC下,block总能正确运行。如果你不用ARC,最好能保证在 stack 内存中声明定义的block,能够拷贝到heap内存,保证block的正常运行。

当然,事情并不是这么简单,看苹果官方文档

Blocks “just work” when you pass blocks up the stack in ARC mode, such as in a return. You don’t have to call Block Copy any more. You still need to use [^{} copy] when passing “down” the stack into arrayWithObjects: and other methods that do a retain.

[译] Block 小测验的更多相关文章

  1. 来试试这个来自静态代码分析工具PVS Studio提供C++的小测验吧

    博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:来试试这个来自静态代码分析工具PVS Studio提供C++的小测验吧.

  2. Java web 小测验

    题目要求: 1登录账号:要求由6到12位字母.数字.下划线组成,只有字母可以开头:(1分) 2登录密码:要求显示“• ”或“*”表示输入位数,密码要求八位以上字母.数字组成.(1分) 3性别:要求用单 ...

  3. BZOJ4635 : 数论小测验

    第一问: 设$a[i]$表示使用$[1,i]$的数字$n$次形成的数组里有多少个$\gcd=1$. 考虑容斥,则$a[i]=i^n-\sum_{j=2}^i a[\lfloor\frac{i}{j}\ ...

  4. oracle12c数据库第一周小测验

    一.单选题(共4题,30.4分) 1 (  )是位于用户与操作系统之间的一层数据管理软件.数据库在建立.使用和维护时由其统一管理.统一控制.   A. A.DBMS B. B.DB C. C.DBS ...

  5. JS小测验

    1.编写一个方法method(),判断一个数能否同时被3和5整除 <div class="one" onClick="method()"> func ...

  6. jQuery小测验

    1.在div元素中,包含了一个<span>元素,通过has选择器获取<div>元素中的<span>元素的语法是? 提示使用has() $(div:has(span) ...

  7. 【OCP、OCM、高可用等】小麦苗课堂网络班招生简章(从入门到专家)--课程大纲

    [OCP.OCM.高可用等]小麦苗课堂网络班招生简章(从入门到专家)--课程大纲 小麦苗信息 我的个人信息 网名:小麦苗 QQ:646634621 QQ群:618766405 我的博客:http:// ...

  8. 3-11 《Ruby元编程》第4章block块 3-12

    第4章代码块blocks 基础知识 作用域:用代码块携带variables through scopes 通过传递block给instance_eval方法来控制作用域. 把block转换为Proc, ...

  9. 贴心小技能——纯CSS实现的帮助提示

    1. 新技能传授---哒哒哒哒 我们经常会接到这样的小需求,鼠标放在某个位置实现一段小提示. 你还在用js实现这样一个小功能,你就太out了,来看看我们用纯CSS打造的帮助提示. 2. html &l ...

随机推荐

  1. unionpay技术服务开放平台

    URL: https://open.unionpay.com/ajweb/index USER: jimingsong PWD: qweasd

  2. 【IE6的疯狂之九】li在IE中底部空行的BUG

    曾经写过[IE6的疯狂之六]li在IE中底部3像素的BUG(增加浮动解决问题),原文地址:http://www.css88.com/archives/421: IE6 BUG大全: http://ww ...

  3. 《Javascript权威指南》

    <Javascript权威指南> chorme.safari中的input或textarea html超链接(a)详细讲解 html5新增及删除标签 html表格 图片加alt属性

  4. C# Oracle insert 中文乱码

    问题描述: 在PL SQL中insert 中文数据,显示不乱码,通过后台insert的中文数据,显示问号. 解决分三步: 1.Select userenv('language') from dual; ...

  5. Mybatis一对多查询得不到多方结果

    一对多查询:一个年级对应多个学生,现在要查询年级(带学生)信息. 查询结果: [main] INFO com.java1234.service.GradeTest - 查询年级(带学生)[main] ...

  6. 推荐几个好的 Maven 常用仓库网址

    注意,以下内容转载自:推荐几个好的 Maven 常用仓库网址 Maven 确确实实是个好东西,用来管理项目显得很方便,但是如果是通过 Maven 来远程下载 JAR 包的话,我宿舍的带宽是4兆的,4个 ...

  7. 防范XSS

    .net framework4.5 提供了AntiXss类,来防范XSS攻击. 在开放指令的同时过滤危险字符串,使用AntiXss.GetSafeHtmlFragment(html)方法,具体可以参照 ...

  8. java栈内存堆内存和GC相关

    java栈内存堆内存 Java把内存分成两种,一种叫做栈内存,一种叫做堆内存,有着不同的作用.栈内存用来存储局部变量和方法调用.栈内存归属于单个线程,每个线程都会有一个栈内存,其存储的变量只能在其所属 ...

  9. C#拾遗(一、基本类型)

    1. C#是一种块结构语言,用花括号{}分块,但是用#region和#endregion来定义可以展开和折叠的代码区域 #region 这是引用区 using System; ...... #endr ...

  10. centos解压bz2文件出错

    出现的问题: 用tar 解压 tar.bz2文件出错 debian:/usr/src# tar jxf linux-2.6.26.tar.bz2tar: bzip2: Cannot exec: No ...