美文网首页
block的本质及clang对block的实现

block的本质及clang对block的实现

作者: 一双鱼jn | 来源:发表于2018-11-08 17:03 被阅读10次

Block的本质

Block转换为C++

这样一段代码

int main(int argc, const char * argv[]) {
    void (^blk)(void) = ^{
        printf("Block");
    };
    blk();
    return 0;
}

经过clang转换为c++代码之后

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int flags=0) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

struct __block_impl {
  void *isa;
  int Flags;
  int Reserved;
  void *FuncPtr;
};

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)};

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {

        printf("Block");
}

int main(int argc, const char * argv[]) {
    void (*blk)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));
    
    ((void (*)(__block_impl *)) ((__block_impl *)blk)->FuncPtr) ((__block_impl *)blk);
    return 0;
}

将代码拆开来看

OC中定义一个Block

    void (^blk)(void) = ^{
        printf("Block");
    };

转换成了下面的代码。

    void (*blk)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));

去掉类型转换,将代码简化

    void(*blk)(void) = __main_block_impl_0(__main_block_func_0, &__main_block_desc_0_DATA)

从上面可以看到,=后面是创建一个Block的部分,转换之后创建Block的过程其实就是调用了__main_block_impl_0结构体的构造方法生成一个结构体的实例。

这个构造方法中传入了两个参数。第一个是__main_block_func_0,从上面的代码可以看到,这是一个函数指针。这个函数就是Block的具体实现转换成的函数。第二个参数是一个结构体实例的指针。这个参数是确定了结构体所需要的空间大小。

通过上面分析可以看到,定义一个block本质是定义了一个结构体实例。

接下来看=左边的部分。

=左边定义了一个Block类型的blk变量。从上面可以看到右边是调用了结构体的构造函数,那么自然生成的是结构体的实例,所以Block类型的变量其实也就是一个结构体的指针。void(^blk)(void) == struct __main_block_impl_0 *blk

block对象转换为c++之后其实就是一个结构体实例,block变量就是一个结构体实例的指针

解析__main_block_impl_0结构体

既然Block就是一个__main_block_impl_0的实例,那么看看这个结构体的具体内容

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int flags=0) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

这个结构体有两个属性和一个构造方法,去掉构造方法之后,这个结构体就只有两个属性。

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
};

第一个属性是一个结构体,这个结构体的声明如下

struct __block_impl {
  void *isa;
  int Flags;
  int Reserved;
  void *FuncPtr;
};

这个结构体定义了一个函数指针的属性,这个指针指向了block实现转换后的函数。

还有一个isa指针。用于指向该block所属的类型

block的转换后代码对比和关系如图所示

block_essence.png

相关文章

  • block的本质及clang对block的实现

    Block的本质 Block转换为C++ 这样一段代码 经过clang转换为c++代码之后 将代码拆开来看 OC中...

  • block

    目录一、block相关概念 1、block的本质 2、block的声明、实现、调用格式及block代码的执行顺序二...

  • iOS中Block实现原理的全面分析

    Block的底层基本结构 通过clang命令查看编译器是如何实现Block的,在终端输入clang -rewrit...

  • Block源码解析和深入理解

    Block源码解析和深入理解 Block的本质 Block是"带有自动变量值的匿名函数". 我们通过Clang(L...

  • 简述Block底层

    Block底层解析 block编译转换结构 对其执行clang -rewrite-objc编译转换成C++实现,得...

  • block学习笔记

    原文发布在个人博客 clang工具 block分类 block 结构 block调用 block类型以及ARC对b...

  • iOS Block简介

    Block的本质 Block是将函数及其执行上下文封装起来的对象。Block调用即是函数的调用。 clang(LL...

  • iOS Block浅析

    Block实现原理 要想知道Block的内部实现,需要知道Block编译完后是什么样子,使用clang可看到Blo...

  • Block学习笔记一之结构探究

    Block代码实现探究 使用clang命令 其中block.m是文件的名字,转换为可读的源代码:block.cpp...

  • Block总结

    一、Block的底层结构及本质 (1)block本质: 从代码可以看出,Block的本质就是NSObject. 也...

网友评论

      本文标题:block的本质及clang对block的实现

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