iOS Block是如何引用外部变量的?

1. block如何引用外部基本数据类型变量?

block内引用基本数据类型,会直接将基本数据类型在内存中复制出另一个新的基本数据类型变量供block使用,相当于对象的深复制。
所以在执行block前改变了基本数据类型变量的值时,不会影响block内部已经进行过“深复制”的值。这里用引号包括深复制一词,是因为深复制一词的含义时对于对象来说的, 这里只是引申了深复制的含义。

// 若在执行block前,改变基本数据类型的值,不改变block内部已经引用的基本数据的值
    inta =10;
    void(^block1)(void) = ^{
        NSLog(@"log: 引用的基本数据a:%d", a);
    };
    a =20;
    block1();
 // 打印结果:
 // log: 引用的基本数据a:10

2. block如何引用外部对象?

block内引用对象, block会通过复制对象的地址来实现访问对象,进行一次强引用,相当于一次浅复制,但由于在block内部不能对对象进行赋值操作,所以相当于强引用了一个readonly的对象。

// block可以调用对象的对象方法
// block内虽然不能对对象进行赋值操作, 但可以调用对象的对象方法。
  NSMutableString * aString = [@"abc" mutableCopy];
  void(^block2)(void) = ^{
    NSLog(@"log:引用了NSMutableString对象 %@", aString);
    [aStringappendString:@"--abc"];
    NSLog(@"log:引用了NSMutableString对象, 并在block内部调用对象的对象方法 %@", aString);
  };
  block2();
// 打印结果:
// log:引用了NSMutableString对象 abc
// log:引用了NSMutableString对象, 并在block内部调用对象的对象方法 abc--abc
// block会强用对象
// 由于block会强引用对象, 所以即使在执行block前, 在block外部将对象指针置为空,   
   对象也不会立即销毁, 在block执行完成后, block才会解除与对象的强引用关系。
// 在上面代码的基础上, 接着这样做:
  aString =nil;
  block2();
// 打印结果: 
// log:引用了NSMutableString对象 abc--abc
// log:引用了NSMutableString对象, 并在block内部调用对象的对象方法 abc--abc--abc
// 在上面代码的基础上,再接着打印aString的值, 这个是才会发现aString被销毁。
  aString =nil;
  NSLog(@"log: 执行完block后的aString的值: %@", aString);
// 打印结果: log: 执行完block后的aString的值: (null)


// 正是由于block会强引用对象的关系, 所以需要避免block造成的循环引用问题。
// block执行完毕后, 对象sString不再被任何一个指针强引用, 所以aString被释放。

3.__block的含义

先解释一下block不被允许修改外部变量值的含义:这里所说的外部变量的值,指的是栈中指针的内存地址。对于对象,对象的指针地址是记录在栈中(对象在堆中),我们不能修改的就是栈中的对象指针。
被block引用的外部变量在block内部是不能进行操作的,所以想要在block内部对外部变量赋值时,需要使用__block去修饰变量。
被__block修饰的对象,block会对对象类型的指针从栈中copy到堆中,这不会改变该指针所指向的堆中的地址。
被__block修饰的基本数据变量,block会将基本数据从栈上直接copy到堆上。

参考自:iOS中__block 关键字的底层实现原理

4.__weak修饰对象

由于block默认强引用对象,可能会造成block与对象间的循环引用,使用__weak弱引用修饰对象可以解决循环引用的问题。但由于weak指针可能会被自动置为空,所以如果block内部需要保证对象存活时,需要在block内部,使用strong对weak指针进行一次强引用,防止weak指针变为空。

你可能感兴趣的:(iOS Block是如何引用外部变量的?)