�面试题1 -- 输出结果是多少?
- (void)blockDemo1
{
int num = 10;
void(^task)() = ^{
NSLog(@"%d",num);
};
num = 20;
task();
}
答案是10
结果分析
- 当在Block内部“访问”外部变量时,Block会对外部的变量进行一次"临时“的”拷贝“;
- 临时拷贝的结果:把栈区的地址拷贝到堆区(就是把这个变量拷贝一份,在堆区创建一个房子存放它。)
- 其实,在Block内部操作的是副本(临时拷贝出来的那一份); 对Block外部的变量的真实值不会造成影响。
其实,在定义代码块时,就已经完成拷贝
void(^task)() = ^{ NSLog(@"===> %p-%d",&num,num); };
- (void)blockDemo1
{
int num = 10;
// => 0x7fff503095bc-10 : 栈区
NSLog(@"栈区地址 => %p-%d",&num,num);
// 其实,在定义代码块时,就已经完成拷贝
// void(^task)() = ^{ NSLog(@"===> %p-%d",&num,num); };
void(^task)() = ^{
// 栈区地址比堆区地址大
NSLog(@"堆区地址 ===> %p-%d",&num,num);
};
num = 20;
// 0x7fff503095bc-20 : 栈区
NSLog(@"栈区地址 ==> %p-%d",&num,num);
task();
}
面试题2 -- 代码是否有错误? 如果有,如何修改?
- (void)blockDemo2 {
int i = 10;
void (^block)() = ^ {
i = 30;
NSLog(@"%d", i);
};
i = 20;
block();
}
答案 : 有
提示 : 默认情况下,block 不允许直接修改外部变量的值
结果分析
- 1.当在Block内部”修改“外部变量,不被允许
- 2.如果非要在Block内部修改外部变量,需要使用__block修饰外部变量
- 3.__block修饰外部变量作用:使外部变量可以在Block内部修改
- 4.被__block标记的外部变量,一旦在Block内部”使用过“,那么Block对外部变量的拷贝就不是临时的了;Block外部变量的真实值就会发生变化,那么这个变量的地址在后续的使用中都是堆区地址
- 5.为什么在Block内部不能访问外部变量?(这个还不能很好的理解)
- block一般是作为回调使用,很多时候需要传递到另外的类里面,局部的变量处理block的作用域就销毁了,为了让局部的变量不销毁,所有就有了
__block
去标记它! - (因为Block设计用来做数据的传递的,Block一般会传递到另外的类里面做回调.如果Block内部的变量在栈区,那么Block在传递的过程中,它内部的变量容易丢失;)
- (void)blockDemo2
{
// 使用__block标记外部变量
__block int num = 10;
// => 0x7fff5430c5b8-10 : 栈区
NSLog(@"=> %p-%d",&num,num);
//在定义代码块时,就已经完成拷贝(并且因为在block里,使用过外部
变量,所以这个拷贝是永久的,在堆里)
void(^task)() = ^{
num = 30;
// ===> 0x600000029bd8-30 : 堆区
NSLog(@"===> %p-%d",&num,num);
};
//那么这个变量的地址在后续的使用中都是堆区地址
num = 20;
// ==> 0x600000029bd8-20 : 堆区
NSLog(@"==> %p-%d",&num,num);
task();
}
面试题3 -- 代码是否有错误?如果有,如何修改?
- (void)blockDemo3
{
NSMutableString *strM= [NSMutableString stringWithString:@"hello"];
void (^block)() = ^ {
[strM appendString:@"hehe"];
};
block();
}
答案 : 没有
提示 : block 内部修改的是 strM 指向堆区中的字符串内容,并没有修改 strM 本身的堆区地址
结果分析
- (void)blockDemo3
{
NSMutableString *strM= [NSMutableString stringWithString:@"hello"];
NSLog(@"strM指针在栈区的地址%p strM内容在堆区的地址%p",&strM,strM);
void (^block)() = ^ {
// 只修改strM指向堆区的内容(字符串);没有修改strM指针所在的地址
//(存放字符串的堆区地址没变,只是&strM变成堆区地址);
[strM appendString:@"hehe"];
NSLog(@"strM指针拷贝到堆区的新地址%p strM内容在堆区的地址%p",&strM,strM);
};
block();
}