block 本质

2018-08-24  本文已影响85人  mark666

一.block底层源码实现

1.定义一个简单的block

        int a = 100;
        void (^block)(void) = ^(){
            NSLog(@"print block");
            NSLog(@"print args %d",a);
        };

2.生成对应的底层代码:

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m

3.摘取主要的代码

struct __block_impl {
    void *isa;
    int Flags;
    int Reserved;
    void *FuncPtr;
};
struct __main_block_desc_0 {
    size_t reserved;
    size_t Block_size;
};
struct __main_block_impl_0 {
    struct __block_impl impl;
    struct __main_block_desc_0* Desc;
    int a;
    // C++ 的构造函数
    __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _age, int flags=0) : age(_age) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static void __main_block_func_0(struct __main_block_impl_0 *__cself){
 int a = __cself->a; 
 NSLog((NSString *)&__NSConstantStringImpl__var_folders_qx_8kn3tsws6hz4j8n03t14prmh0000gn_T_main_8dfc00_mi_0);
 NSLog((NSString *)&__NSConstantStringImpl__var_folders_qx_8kn3tsws6hz4j8n03t14prmh0000gn_T_main_8dfc00_mi_1,a);
 }
static struct __main_block_desc_0 {
  size_t reserved;
  size_t Block_size;
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0)};

int a = 100; 
//定义block变量
void (*block)(void) = &__main_block_impl_0(
__main_block_func_0, 
&__main_block_desc_0_DATA, 
a);
// block 的调用
block->FuncPtr(block);

从以上底层代码中我们可以发现,当我们声明一个block 对应的会转换为底层c++__main_block_impl_0 结构体,这个结构体传入对应的参数

得出结论:

这样的话,我们就可以通过 void (*block)(void) 来接收这个结构体的地址,通过调用这个 FuncPtr 来调用函数。

关系图

二.block 的变量捕获

变量类型 捕获到block内部 访问方式
局部变量auto ✔️ 值访问
局部变量static ✔️ 指针传递
全局变量 直接访问

auto 离开作用域就会销毁

全局变量不会被捕获,局部变量会被捕获,self 属于局部变量

三.block类型

block 有三种类型,可以通过调用class方法或者isa指针查看具体类型,最终都继承自NSBlock

block类型 环境
NSGlobalBlock 没有访问auto变量
NSStackBlock 访问auto变量
NSMallocBlock NSStackBlock 调用了copy
// 没有访问auto变量
// 访问static 和 全局变量 都是  __NSGlobalBlock__类型
void (^block)(void) = ^(){
            NSLog(@"print block");
        };

注意ARC 环境问题

Block的类 副本源的配置存储域 复制效果
_NSConcreteStackBlock 从栈复制到堆
_NSConcreteGlobalBlock 程序的数据区域 什么也不做
_NSConcreteMallocBlock 引用计数增加

四、对象类型的auto变量

1.当block内部访问了对象类型的auto变量时

2.如果block被拷贝到堆上

如果block从堆上移除

函数 调用时机
copy函数 栈上的Block复制到堆时
dispose函数 堆上的Block被废弃时

使用clang 转换OC为C++代码时,遇到

__weak问题
cannot create __weak reference in file using manual reference

解决方案:支持ARC 指定运行时系统版本

Xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc -fobjc-arc -fobjc-runtime=ios-8.0.0 main.m

五、 __block修饰符到底干了什么

1. __block 修饰符

2. 探究__block 底层干了什么

修饰基本数据类型

   __block int a = 8;
        void (^block)(void) = ^{
            a = 10;
            NSLog(@"------");
        };
        
        block();
        NSLog(@"-----%d",a);

首先经过__block 修饰的 int a 会被转换为一个结构体__Block_byref_a_0 a,并将给这个结构体赋值

struct __Block_byref_a_0 {
  void *__isa;
__Block_byref_a_0 *__forwarding;
 int __flags;
 int __size;
 int a;
};

__Block_byref_a_0 a = {
                         0,
                         &a,
                         0,
                         sizeof(__Block_byref_a_0),
                        8};

接下下来在void (^block)(void) 中给 a 赋值,通过 a->__forwarding->a,即:

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  __Block_byref_a_0 *a = __cself->a;
   (a->__forwarding->a) = 10;
}

结论:

修饰对象类型

六、block 的内存管理

对象类型的auto变量、__block变量

MRC环境下的内存管理情况

1.__block 修饰时

__block Person *person =  [ [Person alloc] init];

void (^block)(void) = [^{
    NSLog(@"%p",person);
} copy];

[person release];

block();

[block release];

2.未进行__block 修饰时

Person *person =  [ [Person alloc] init];

void (^block)(void) = [^{
    NSLog(@"%p",person);
} copy];

[person release];

block();

[block release];

七、__block 的 __forwarding指针

__forwarding

__forwarding 指针保证了可以更改__block 修饰的变量,不论这个block是在栈上,还是在堆上。

八、解决循环引用方式

MRC 情况下 __block 不会对外部的变量产生强引用

九、总结

ARC 下 修饰block 使用 strongcopy 没有区别,都会将栈上的拷贝到堆上
MRC 下 strong 强引用 copy 会将栈上的拷贝到堆上

上一篇 下一篇

猜你喜欢

热点阅读