为什么我选择使用 Blocks(块)

  • 扯淡:到了新公司接手新框架之后,发现大量的使用Blocks,之前很多时候都是使用代理,突然面对这个陌生的语法,特地科普总结了一番。


什么是Blocks

  • 一句话概括就是,带有局部变量的匿名函数(即不带名称的函数)。也称为闭包。

Blocks语法

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

//例如

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

上述表达式中,返回值类型 以及 参数列表可以省略(如下), 语法加粗表明不可缺少的部分。

^ {

printf("Hello, blocks");

}


将Blocks声明为属性

  • 多数情况下,我们会把Blocks声明为属性,便于调用。可以利用关键字typedef 对Blocks进行别名。

typedef 返回值类型 (^Blocks别名)(参数列表);

typedef int (^blk_t)(int);

//将Blocks声明为属性

@property (nonatomic, copy) blk_t blk;

//对Blocks进行声明定义

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

//调用Blocks 和 调用C语言声明函数方式一样

NSLog(@"%d\n", blk(10));

将Blocks作为参数传入方法中

  • 通常都会事先为Blocks取一个别名,一来方便理解,好吧不装逼,就是偷懒。

//定义:

typedef void(^Block)(int a);

- (void)nslogParameterWithBlock:(Block)block {

if (block) {

block(10);

}

}

//赋值并调用

[object nslogParameterWithBlock:^(int a) {

NSLog(@"%d", a);

}];


Blocks截获自动变量值

  • 什么是截获自动变量值,所谓自动变量,就是局部变量。那为什么会截获呢?先来看个例子。

int val = 10;

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

val = 20;

block();  //此处调用block,打印为val = 10;

  • 之所以没有打印val = 20,是因为该Blocks在编译期就进行了初始化,对声明的val进行截获,也就是说,把val拷贝一份存入Blocks中。所以当外部val进行了修改时,不影响Blocks截获的val值。


__block 关键字

  • 从上个例子中,我们知道Blocks会对变量进行截获。那如果我非要在Blocks中修改外部参数怎么办呢?我们可以通过__block关键字做到!

__block int val = 10;

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

val = 20;

block();  //此处调用block,打印为val = 20;

  • 通过__block说明符对外部参数进行声明后,实际上,相当于在声明定义Blocks的时候,不是通过拷贝内存的方式拷贝外部参数,而是通过指针指向该外部变量,从而达到可以在Blocks中修改的效果。


总结:

  • Blocks使用方便,但是block的内存管理确实是一项非常重要也非常容易出错的地方,稍不注意便会造成内存泄露,所以,在使用Blocks的时候,也要小心。

  • 因为是匿名函数,所以在项目中多人合作的时候,最后写上每个Blocks的注释说明。

  • 后续将继续总结Blocks的内存管理内容。

上一篇:刷题随记 - 8. 构造最大二叉树(654)


下一篇:Java基础总结--流程控制