《Objective-C 高级编程》第一篇:Block概要

本系列文章主要是对《Objective-C 高级编程》这本书做的读书笔记总结,除了这本书中的内容以外,也加上了自己对开发技术的理解和一些个人的经验分享。

一、什么是Blocks

1.1 Blocks概要

Blocks是C语言的扩充功能,是带有局部变量的匿名函数(匿名其实就是没有名称的函数,C语言标准不允许匿名函数)

例如:这里声明了一个名称为func的函数

int func (int count);

为了调用该函数,必须使用该函数的名称

int  result = func (10);

如果像下面这样,使用函数指针来代替直接调用函数,那么不需要使用函数名也能够使用该函数

int  result = (*funcPtr) (10);

但其实使用函数指针也仍然需要知道函数名称。例如以下,在赋值给函数指针时,若不使用函数的名称,就取法取得该函数的地址。

int (*funcPtr)(int) = &func;
int  result = (*funcPtr) (10);

而通过Blocks,源代码中就能够使用匿名函数。

匿名函数大家已经知道了,那么现在让我们来看一下C语言中可能使用的变量有哪些 :

  • 局部变量
  • 函数参数
  • 静态全局变量
  • 静态局部变量
  • 全局变量

1.2 Blocks语法

下面我们来详细讲解一下带有局部变量值的匿名函数Blocks的语法:

^ 返回值类型 参数列表 表达式

以下定义表明这是一个表示没有返回类型,并且参数为int型的Block
^ void (int a, int b) {
  // do sth
}

如上所示:完整形式的Blocks和C语言的函数定义区别为:

  • 没有函数名 (匿名函数)
  • 没有 ^

温馨小贴士:因为OS X,iOS 应用程序的源代码中大量使用Block,所以插入该记号便于查找。

block可以省略返回类型

省略返回类型的语法为: ^ 参数列表 表达式

^ void (int a, int b) {表达式}

等价于

^ (int a, int b) {表达式}

其次,如果不使用参数,block也可以省略参数列表

语法为: ^ 表达式

^ {
  // 说点什么吧,少年
}

1.3 Block与C函数对比

int func (int count) 
{
  return count + 1;
}
// 函数func的地址赋值给函数指针变量funcPtr
// 在block语法下可将block赋值给声明为block类型的变量
int (*funcPtr) (int) = &func
int (^blockName) (int)

与前面的使用函数指针的源代码对比可知,声明block类型变量仅仅是将声明函数指针类型变量 * 变为 ^

int (^block) (int) = ^ (int count) {
    return count + 1;
}

block类型变量与c语言变量相同,block也可以作为函数参数传递或者函数的返回值

  • 作为函数参数
void func ( int (^block) (int) ) {
  //
}
  • 作为函数的返回值
int (^func()) (int) {
  //
  return ^ (int count) {
      return count + 1;
  };
}

由上面源代码可以看出在使用block变量时,记录方式非常复杂。我们可以像使用函数指针类型那样,使用 typedef来解决

  typedef int (block_t) (int)

我们来对比以下

// 没有定义前
void func ( int (^block) (int) ) {

}

// 定义后
void func (block_t block) {

}

另外,Block调用 和 C语言中使用函数指针调用函数的方法几乎完全相同。

int result = (*funcPtr)(10);
int result = block(10);

也可以使用指向block类型变量的指针调用block

typedef int (^block_t) (int)

block_t block = ^ (int count) {
  return count + 1;
}

block_t *blockPtr = █

(* blockPtr)(10);

1.4 截获局部变量值以及__block说明符的使用

1.4.1 截获局部变量值

截获局部变量值是指保存执行block语法瞬间的值,并且保存后就不能修改变量值。

int val = 0;

void (^block) (void) = ^ {
  NSLog(@"val = %d", val);
}

val = 1;

block();

执行上面源代码,打印val的值为0。这是因为在block中截获了局部变量的值,即保存了该变量的瞬间值。所以即使更改了变量的值也不会影响block的打印。

1.4.2__block说明符的使用

执行下面源代码,会产生编译错误。

int val = 0;

void (^block) (void) = ^ {
  val = 1;
}

block();

NSLog(@"val = %d", val);

向截获的变量直接赋值会发生编译错误,但使用截获的值却不会报错。

id array = [NSMutableArray array];

void (^block) (void) = ^ {
  id obj = [[NSObject alloc] init];
  [array addObject:obj];
}

block();

若想在block中修改局部变量的值,需要在该自动变量前加 __block 说明符。

__block int val = 0;

void (^block) (void) = ^ {
  val = 1;
}

block();

NSLog(@"val = %d", val);

源代码的执行结果为:val = 1

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,776评论 0 9
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,958评论 19 139
  • 当年流行的美剧CSI迈阿密第10季里,前瞻性的出现过一款透明手机,在剧中与会拐弯的子弹一起成了观众的怨念。在CW台...
    2平方单人床阅读 554评论 0 0
  • 创业公司不需要产品经理 近期帮朋友公司管理整体公司,又接触了一线的产品技术、市场、商务等等,让我不禁想写点东西,分...
    舒星阅读 875评论 0 18
  • 总是把开头想得那么难 把过程想得那么简单 却没想到结局那么悲惨 时间在六月的炽阳下翻滚挣扎 蝉鸣着一声两声是要离开...
    不迩阅读 453评论 0 1