Block

作者: 没戏还在演戏 | 来源:发表于2023-01-30 20:45 被阅读0次

在iOS中,block编程使用得很频繁,我们不仅要会用block,更需要理解block的底层实现原理。笔者在面试中,block问题是必问的。

什么是block

block是iOS中对闭包的实现,什么是闭包呢?闭包(英语:Closure),又称词法闭包(Lexical Closure)或函数闭包(function closures),是在支持头等函数的编程语言中实现词法绑定的一种技术。闭包在实现上是一个结构体,它存储了一个函数(通常是其入口地址)和一个关联的环境(相当于一个符号查找表)。环境里是若干对符号和值的对应关系,它既要包括约束变量(该函数内部绑定的符号),也要包括自由变量(在函数外部定义但在函数内被引用),有些函数也可能没有自由变量。

block类型

block是一个OC对象,block类型有NSStackBlockNSMallocBlockNSGlobalBlock、,分别分配在栈、堆、全局存储区域中。他们都继承于NSObject。下面代码证明打印了NSGlobalBlock的继承链

void (^block)(void) =  ^{
        NSLog(@"akon");
    };

    NSLog(@"block.class = %@", [block class]);
    NSLog(@"block.class.superclass = %@", [[block class] superclass]);
    NSLog(@"block.class.superclass.superclass = %@", [[[block class] superclass] superclass]);
    NSLog(@"block.class.superclass.superclass.superclass = %@", [[[[block class] superclass] superclass] superclass]);

运行结果为:
2020-11-13 18:39:02.919351+0800 BlockTestDemo[86009:2083840] block.class = __NSGlobalBlock__
2020-11-13 18:39:02.919562+0800 BlockTestDemo[86009:2083840] block.class.superclass = NSBlock
2020-11-13 18:39:02.919713+0800 BlockTestDemo[86009:2083840] block.class.superclass.superclass = NSObject
2020-11-13 18:39:02.923424+0800 BlockTestDemo[86009:2083840] block.class.superclass.superclass.superclass = (null)

下面表格列出了MRC和ARC环境下block类型

MRC下block类型

类型 环境
NSGlobalBlock 只访问了静态变量(包括全局静态变量和局部静态变量)和全局变量
NSStackBlock 没访问静态变量和全局变量
NSMallocBlock NSStackBlock调用了copy

执行如下代码,打印结果符合预期

 __weak typeof(self)weakSelf = self;

    static int a = 0;
    void (^block1)(void) =  ^{
        a = 1;
        b = 1; //b为全局变量

    };

    __block int c = 0;
    void (^block2)(void) =  ^{
        NSLog(@"age:%d", weakSelf.age);
        c = 1;
    };

    NSLog(@"block1.class = %@", [block1 class]);
    NSLog(@"block2.class = %@", [block2 class]);
    NSLog(@"block2 copy.class = %@", [[block2 copy] class]);

运行结果如下:
2020-11-14 22:45:54.457496+0800 BlockTestDemo[13178:426318] block1.class = __NSGlobalBlock__
2020-11-14 22:45:54.457616+0800 BlockTestDemo[13178:426318] block2.class = __NSStackBlock__
2020-11-14 22:45:54.457720+0800 BlockTestDemo[13178:426318] block2 copy.class = __NSMallocBlock__

ARC下block类型

类型 环境
NSGlobalBlock 只访问了静态变量(包括全局静态变量和局部静态变量)和全局变量
NSMallocBlock 没访问静态变量和全局变量

运行上面的代码,结果如下:

2020-11-14 22:45:54.457052+0800 BlockTestDemo[13178:426318] block1.class = __NSGlobalBlock__
2020-11-14 22:45:54.457211+0800 BlockTestDemo[13178:426318] block2.class = __NSMallocBlock__
2020-11-14 22:45:54.457356+0800 BlockTestDemo[13178:426318] block2 copy.class = __NSMallocBlock__

ARC下自动copy

  • 我们看到block2为NSMallocBlock,这是因为编译器做了优化,在ARC下除了NSGlobalBlock_就是NSMallocBlock,没有NSStackBlock;在MRC NSMallocBlock生成的条件是对block调用了copy操作。
  • 在ARC环境下,编译器会根据情况自动将栈上的block复制到堆上,copy的情况如下:
    1、block作为函数返回值时
    2、 将block赋值给_strong指针时
    3、block作为Cocoa API中方法名含有usingBlock的方法参数时
    4、block作为GCD API的方法参数时
    在ARC中对NSStackBlock调用copy变成NSMallocBlockNSMallocBlock调用copy还是NSMallocBlock,引用计数+1,NSGlobalBlock
    调用copy啥都不做。
  • copy底层原理
    1、通过_Block_object_assign来对OC对象进行强引用或弱引用
    2、通过_Block_object_dispose对OC进行清理

block数据结构和变量捕获

在Block内部无法修改外部数据的原因是在Block中访问外部变量时,都会对其进行一份拷贝,需要注意这里的拷贝是直接拷贝,如果你在Block内部和外部对变量对象进行打印,则可以看到其地址是不同的。Objective-C提供了_block关键字,使用它可以直接访问原始变量。

变量捕获

可以按照上面分析思路,得出结论

变量类型 捕获到block内部 变量类型
局部非OC变量 值传递
局部变量 static、OC对象 指针传递
全局变量 × 直接访问

可以看到全局变量,b
lock内部不会直接捕获,其他变量会捕获。

__block变量

__block作用

  • __block只能修饰非静态局部变量,不能修饰静态变量和全局变量,否则编译器报错。
  • 当需要在block内部修改一个局部变量时,需要加__block ,否则,编译不过。下面的代码,编译报错:Variable is not assignable (missing __block type specifier)。加上__block编译通过,name会变成lbj
 NSString* name = @"akon";
    void (^block)(void) =  ^{
        name = @"lbj";
     };

    block();

底层实现

  • 类似刚才的转成cpp思路,分析得出结论如下图。总结就是对于__block变量,底层会封装成一个对象,其中通过__forwarding指向自己,来访问真实的变量。


    image
  • 为什么要通过__forwarding访问?
    这是因为,如果__block变量在栈上,就可以直接访问,但是如果已经拷贝到了堆上,访问的时候,还去访问栈上的,就会出问题,所以,先根据__forwarding找到堆上的地址,然后再取值

循环引用

循环引用原因

当对象A和对象B互相引用时会造成循环引用。

循环引用解决方案

竟然对象A和对象B互相引用会造成循环引用,那就要断开这个循环引用,可以通过__weak或者__unsafe_unretained,这两者的区别是__unsafe_unretained当引用对象变为nil时__unsafe_unretained对象不会自动置为nil,导致变为野指针,再次使用会崩溃。

常见循环引用及解决

1) 在VC的cellForRowAtIndexPath方法中cell的block直接引用self或者直接以_形式引用属性造成循环引用。

 cell.clickBlock = ^{
        self.name = @"akon";
    };

cell.clickBlock = ^{
        _name = @"akon";
    };

解决方案:把self改成weakSelf;

__weak typeof(self)weakSelf = self;
    cell.clickBlock = ^{
        weakSelf.name = @"akon";
    };

注意有的时候我们会在block里面写成__strong typeof(weakSelf) strongSelf = weakSelf,然后再用strongSelf调用方案,这样做的原因是防止在block执行过程中weakSelf突然变成nil。
2)在cell的block中直接引用VC的成员变量造成循环引用。

//假设 _age为VC的成员变量
@interface TestVC(){

    int _age;

}
cell.clickBlock = ^{
       _age = 18;
    };

解决方案有两种:

  • 用weak-strong dance
__weak typeof(self)weakSelf = self;
cell.clickBlock = ^{
      __strong typeof(weakSelf) strongSelf = weakSelf;
       strongSelf->age = 18;
    };

  • 把成员变量改成属性
//假设 _age为VC的成员变量
@interface TestVC()

@property(nonatomic, assign)int age;

@end

__weak typeof(self)weakSelf = self;
cell.clickBlock = ^{
       weakSelf.age = 18;
    };

3)delegate属性声明为strong,造成循环引用。

@interface TestView : UIView

@property(nonatomic, strong)id<TestViewDelegate> delegate;

@end

@interface TestVC()<TestViewDelegate>

@property (nonatomic, strong)TestView* testView;

@end

 testView.delegate = self; //造成循环引用

解决方案:delegate声明为weak

@interface TestView : UIView

@property(nonatomic, weak)id<TestViewDelegate> delegate;

@end

4)在block里面调用super,造成循环引用。

cell.clickBlock = ^{
       [super goback]; //造成循环应用
    };

解决方案,封装goback调用

__weak typeof(self)weakSelf = self;
cell.clickBlock = ^{
       [weakSelf _callSuperBack];
    };

- (void) _callSuperBack{
    [self goback];
}

5)block声明为strong
解决方案:声明为copy
6)NSTimer使用后不invalidate造成循环引用。
解决方案:

  • NSTimer用完后invalidate;
  • NSTimer分类封装
+ (NSTimer *)ak_scheduledTimerWithTimeInterval:(NSTimeInterval)interval
                                         block:(void(^)(void))block
                                       repeats:(BOOL)repeats{

    return [self scheduledTimerWithTimeInterval:interval
                                         target:self
                                       selector:@selector(ak_blockInvoke:)
                                       userInfo:[block copy]
                                        repeats:repeats];
}

+ (void)ak_blockInvoke:(NSTimer*)timer{

    void (^block)(void) = timer.userInfo;
    if (block) {
        block();
    }
}

--

怎么检测循环引用

  • 静态代码分析。 通过Xcode->Product->Anaylze分析结果来处理;
  • 动态分析。用MLeaksFinder(只能检测OC泄露)或者Instrument或者OOMDetector(能检测OC与C++泄露)。

相关文章

  • iOS开发之Block原理探究

    Block概述 Block本质 Block调用 Block分类 Block循环引用 Block原理探究 Block...

  • block的使用

    定义block 返回类型 (^block名称)(参数) = ^(){block内容}; 调用block block...

  • Block 02 - __block

    Block 02 - __block __block 的作用 __block 可以解决 Block 内部无法修改 ...

  • iOS面试之Block大全

    Block Block内容如下: 关于Block 截获变量 __block修饰符 Block的内存管理 Block...

  • iOS面试之Block模块

    Block Block内容如下: 关于Block 截获变量 __block修饰符 Block的内存管理 Block...

  • iOS Block

    Block的分类 Block有三种类型:全局Block,堆区Block,栈区Block 全局Block 当Bloc...

  • iOS block 为什么官方文档建议用 copy 修饰

    一、block 的三种类型block 三种类型:全局 block,堆 block、栈 block。全局 block...

  • iOS开发block是用copy修饰还是strong

    Block分为全局Block、堆Block和栈Block1、在定义block没有引用外部变量的时候,block为全...

  • block 初探

    全局block, 栈block, 堆block

  • Block

    一、Block本质 二、 BlocK截获变量 三、__block 修饰变量 四、Block内存管理 五、Block...

网友评论

    本文标题:Block

    本文链接:https://www.haomeiwen.com/subject/xleylrtx.html