首先明确一下Python闭包和ObjC中block的定义
在python中闭包是这么定义的——内层函数引用了外层函数的变量(参数也算变量),然后返回内层函数的情况,称为闭包(Closure);
ObjC中block的定义——Block是一种语言级别的特性,可以允许你创造一些特定的代码块,并像一个普通变量一样让它在方法或者函数中传递。Block也是一种ObjC的对象,也可以添加进入NSArray或者NSDictionary。同时Block还有在它的封闭作用域内捕获值的能力,就像其他语言的闭包或者lambdast式。
从block的定义就可以看出,block和闭包具有很多相同点。接下来我们就来看看block和Python中的闭包的相同点。在Python中,函数本身就可以看作一个变量,而函数的名称就是变量名,所以在python中函数可以作为另外一个函数的参数,当然也就可以作为另外一个函数的返回值。这一点和block一致,一个block也可以作为一个函数的参数,也可以作为一个函数的返回值。还有就是闭包和block都可以在捕获局部变量的值,这是它们的相同点,但是也是它们有差别的地方。先看下面这段ObjC代码:
-(void)testMethod
{
NSInteger aninteger=42;
void(^testBlock)(void)=^{
NSLog(@"Integer is :%ld",aninteger);
};
aninteger=84;
testBlock();
}
在这个ObjC函数中的最后一行中调用testBlock,最后的打印结果不是84,而是42;
在testMethod中定义的testBlock()捕获了局部变量 aninteger,然而在aninteger发生变化后,testBlock中捕获到的aninteger仍然没有发生变化,打印的结果仍然是42,就好比testBlock()捕获aninteger是只是给aninteger照了一个相,然后testBlock()把这张照片保留了下来。当aninteger去整容了(变成了84),再去看block中的相片,相片还是不会变的。所以打印的结果还是42。
在看下面的这段Python的代码:
def testMethod():
i=1
def func():
return i*i;
i=2
return func();
f=testMethod();
print f
最后的打印结果为4:
这段代码func捕获了testMethod中定义的局部i ,当i发生变化的时候,func()捕获到的局部变量也发生了变化,这就和ObjC中block照了一张相不一样,func() 直接把整个人留了下来,你去整容了(i变为了2),人(i)也就变了。
那么怎么让ObjC的blokc也能将整个人留下呢,再看下面的代码。
-(void)testMethod
{
_ _block NSInteger aninteger=42;
void(^testBlock)(void)=^{
NSLog(@"Integer is :%ld",aninteger);
};
aninteger=84;
testBlock();
}
只需要在你要捕获的变量前加上关键字:_ _Block,就可以让block把整个人留下了。
这只是个人的鄙见,有错误还望指正。