1、什么是blocks:带有自动变量(局部变量)的匿名函数。
这里有两个概念:自动变量,匿名函数。
1.1)匿名函数(不带名的函数)
我们来看下函数指针的概念:
int func(int count);
int (*funcptr)(int) = &func; // 函数指针
int result = (*funcptr)(10); // 调用
可以看出上面func这个函数名是多余的,那我们可以去掉他吗,这就要用到匿名函数。也就是blocks的概念。
自动变量在下面会讲到。
2、blocks语法
格式:^ 返回值类型 参数列表 表达式
其中画横线的两个值都是可以省略的。
例子:^int (int count){return count+1;}
3、blocks变量
如上我们所说函数指针可以赋值给一个变量,同样blocks也可以赋值给变量。
这是一个blocks函数:^int (int count){return conunt+1;}
赋值给blocks变量:int(^blk)(int) = ^int (int count){return conunt+1;}
注意:blocks变量声明的形式和函数指针很像。只是把*变成^。
既然blocks变量是一个“变量”,那么就拥有变量相应的用法。可以当作返回值、参数...
当作返回值:
int (^func()) (int)
{
return ^(int count){return count+1;};
}
当作参数:
void func (int (^blk)(int)){.....}
上面就是blocks的基本用法,但是我们不会每次都输入int(^blk)(int) 这么长的变量吧,这也太长了吧,也不好记住呀。下面用typedef改个名称。
typedef int (^blk_t) (int); //这样就可以了,blk_t =》int(^blk)(int)
上面代码简写一下:
void func (blk_t blk){...}
4、自动变量的概念(局部变量)
int main()
{
int dmy = 256;
int val = 10;
const char *fmt = "val = %d \n";
void (^blk)(void) = ^{printf(fmt,val);}
val = 2; // 改变val值
fmt = "这些值改变为:val = %d\n";
blk(); // 调用blocks
return 0;
}
猜一下上面的值输出是10还是2呢? 结果是val = 10
这就涉及到一个概念“截获自动变量”。在blocks表达式里面保存的是该值的“瞬间值”,就像是照相机一样,在那一瞬间保存了该变量,后面就算改写了也不会影响blocks里面的值。
我们把blocks改写成这样:
void (^blk)(void) = ^{ val = 1; }
这样是会报错的,我们不能改变该瞬时值。那么要怎么改呢。可以用到__block说明符。
改成这样:就不会报错了。
__blcok int val = 10;
void (^blk)(void) = ^{ val = 1; }