block块
块对象(block Object)是在Mac OS X10.6和iOS 4.0平台下可以使用的功能,它不是Object-c而是C语言的功能的实现。Apple的文档中将其称为块对象或Block(复数为Blocks),在其他编程语言中,它与闭包(closure)功能基本相同。
块对象的定义
^(参数列){主体}
这里从“^”开始到参数列,主体最后的大括号,这一段记述称为块对象的块句法(block literal)。实际上,块句法并不是被用于在内存中分配的块对象,它只是编写代码时的一种表达用语。
块对象的实例和生命周期
下面是我看到的一个很有趣的例子
#include <stdio.h>
#include <Block.h>
void pr(int (^block)(void))
{
printf("%d
",block());
}
int (^g)(void)=^{return 100;};
void func1(int n)
{
int (^b1)(void)=^{return n;};
pr(b1);
g=b1;
}
void func2(int n)
{
int a=10;
int (^b2)(void)=^{return n*a;};
pr(b2);
}
int main(int argc, const char * argv[])
{
pr(g);
func1(5);
func2(5);
pr(g);
return 0;
}
这是输出的结果
很明显发现最后的pr(g)输出的结果不对
这是因为在func1中将变量b1的块赋值给了外部变量g。但是变量b1中保存的是指向栈的指针。函数func1执行完之后,该块的生命周期也随之结束。由于函数func2的调用会使得栈上被写入其他信息,因此在执行的时候最后pr(g)出错了。
如何修改
将func1中的g=b1;修改为g=Block_copy(b1);,当然要加上头文件 Block.h
Block_copy(block):参数为栈上的块对象,返回堆上复制的块对象。否则(参数为静态数据区或未堆上的块对象)则不进行复制而直接将参数返回,但会增加参数的块对象的引用次数
与之对应的有
Block_release(blcok):减少参数的块的引用计数,减到0时释放块对象的内存区域
对块句法生命周期的总结
- 块句法写在函数外面时,只在静态数据区分配一片内存区域给块对象。这片区域在程序执行期会一直存在。
- 块句法写在函数内时,和自动变量一样,块对象的内存区域会在执行包含块对象的函数时被保存在栈上。该区域的声明周期就是在函数运行期间。