Block不允许修改外部变量的值,这里所说的外部变量的值,指的是栈中指针的内存地址。
__block 所起到的作用就是只要观察到该变量被 block 所持有,就将“外部变量”在栈中的内存地址放到了堆中。进而在block内部也可以修改外部变量的值。
block默认的是NSGlobalBlock类似于函数,存放在代码段;当block内部使用了外部的变量时,block的存放位置变成了NSMallockBlock(堆),所以用__block修饰后才可以在block内部直接修改该变量
block也属于“函数”的范畴,变量进入block,实际就是已经改变了作用域
****************************************************************************
之前学习block的时候告诉我们在其内部是不能修改外部局部变量的值,如果非要修改就要在变量前面加__block,于是我们就一直这么做了,但为什么加上修饰就可以了呢,这个问题今天我决定好好研究一下。
首先看一下Block的分类
NSGlobalBlock:全局block,只要内部没使用外部变量就是全局block。
NSStackBlock和NSMallocBlock:当内部使用外部变量就会变成栈block,当将block从栈去拷贝到堆区就是变成堆block。
首先看一段代码
int a = 0;
1 NSLog(@"a --%p",&a);
void (^test)(void) = ^{
2 NSLog(@"a --%p",&a);
};
3 NSLog(@"a --%p",&a);
test();
代码的输出顺讯为1-3-2,输出结果
2017-10-16 09:49:23.171 fsafdafasf[3674:471719] a --0x7fff4ff9aa2c
2017-10-16 09:49:23.171 fsafdafasf[3674:471719] a --0x7fff4ff9aa2c
2017-10-16 09:49:23.172 fsafdafasf[3674:471719] a --0x600000051cc0
从输出结果可以看出block内和block外是两个作用域,内部的指针a和外部的指针a虽然名字一样但他们的内存地址已经不同了,这也就是我们可以内部修改局部变量的原因。在block内部会将局部变量的指正复制一份,这样内部和外部就不是一个指针了,就算你重新给内部指针赋值,也无法改变外部指针的指向,他们只是名字相同。
接下来再看一段代码
__block int a = 0;
NSLog(@"a --%p",&a);
void (^test)(void) = ^{
NSLog(@"a --%p",&a);
};
NSLog(@"a --%p",&a);
test();
输出结果 1-3-2
2017-10-16 09:58:15.551 fsafdafasf[3711:476037] a --0x7fff5de71a28
2017-10-16 09:58:15.552 fsafdafasf[3711:476037] a --0x6000000283d8
2017-10-16 09:58:15.552 fsafdafasf[3711:476037] a --0x6000000283d8
结论
没有通过__block修饰则是将值传进去,而通过__block修饰之后是将变量的内存地址传进去,这样就可以修改变量的值。