iOS开发 - Block相关的面试题

�面试题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();
}

你可能感兴趣的:(iOS开发 - Block相关的面试题)