美文网首页
Block的底层实现(一):无截获变量Block

Block的底层实现(一):无截获变量Block

作者: 静谧的橘子 | 来源:发表于2017-10-25 08:49 被阅读5次

Blocks用一句话概括就是带有自动变量的匿名函数.

首先先创建一个简单的无参数无返回值的block。

int main(){

void (^blk)(void) = ^{

printf("Block\n");

};

blk();

return 0;

}

在main函数中创建一个简单的block,block中是一个简单的printf;

然后通过clang(LLVM编译器)转换为我们可读源代码的功能;

通过命令clang -rewrite-objc 源代码文件名

然后获得了转换后的Block语法。

转换后的源代码文件有数百行的代码,不用管其他,我们只需要翻到文件的最下边,也就是下图中的位置:

block编译后的代码.png

首先,先寻找到源代码中block的位置,也就是main函数所在的位置,图中可以清晰的看到,然后来看main函数中的代码:

编译后的代码:

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

接着我们来简化一下,去掉转换的部分,具体如下:

struct __main_block_impl_0 tmp = __main_block_impl_0(__main_block_func_0,&__main_block_desc_0_DATA);

struct __main_block_impl_0 blk = &tmp;

这段代码等于__main_block_impl_0结构体实例的指针变量赋给变量blk,

也就是说源代码中的Block就是__main_block_impl_0结构体指针产生的自动变量。

接下里看一下这段代码:

__main_block_impl_0(__main_block_func_0,&__main_block_desc_0_DATA);

第一个参数是由Block语法转换的的C语言函数指针,第二个参数是作为静态全局变量初始化的__main_block_desc_0的结构体指针。

首先看第一个参数的这个函数指针:

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {

printf("Block\n");

}

通过这段代码可以看出,Blocks使用的匿名函数实际上被作为了简单的C语言函数来处理了;

然后看一下传的参数struct __main_block_impl_0 *__cself;

这个__main_block_impl_0结构体的指针__cself相当于Objectice-C中的self;

然后是struct  __main_block_impl_0这个结构体的声明,如下:

struct __main_block_impl_0 {

struct __block_impl impl;

struct __main_block_desc_0* Desc;

}

然后分别看一看其中的结构体,首先是第一个成员变量impl,来看一下__block_impl结构体的声明。

struct  __block_impl{

void * isa;

int Flags;

int Reserved;

void * FucnPtr;

};

从其名称可以看出某些标志、今后版本升级所需的区域以及函数指针。

然后是__main_block_desc_0结构体的声明;

static struct __main_block_desc_0{

size_t reserved;

size_t Block_sized;

};

这些也如同其成员名称所示,其结构为今后版本升级所需区域和block的大小。

接下来就要看一看初始化含有这些结构体的__main_block_impl_0结构体的构造函数。

__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;

}

以上就是无变量截获无返回值的block初始化的源代码。

相关文章

  • Block的底层实现(一):无截获变量Block

    Blocks用一句话概括就是带有自动变量的匿名函数. 首先先创建一个简单的无参数无返回值的block。 int m...

  • Block的底层实现(一):无截获变量Block

    Blocks用一句话概括就是带有自动变量的匿名函数.首先先创建一个简单的无参数无返回值的block。 在main函...

  • Block

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

  • Block(匿名函数)

    一、语法格式 二、截获自动变量值和__block修饰符 block的自动变量截获只针对block中使用的自动变量。...

  • 2022-04-13 block笔记

    什么是block 什么是block调用 block截获变量 __block修饰符用在什么场景,__block变量_...

  • iOS面试之Block大全

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

  • iOS面试之Block模块

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

  • Block的底层实现(二):截获变量的Block

    接着上一篇接下来,添加一下参数看看 通过之前的clang命令编译一下,得到以下的代码 屏幕快照 2017-08-2...

  • Block的底层实现(二):截获变量的Block

    接着上一篇接下来,添加一下参数看看 通过之前的clang命令编译一下,得到以下的代码 首先看一下相同的部分,从上面...

  • iOS block详细知识点

    Block与外界变量 1、截获自动变量(局部变量)值 (1)默认情况 对于 block 外的变量引用,block ...

网友评论

      本文标题:Block的底层实现(一):无截获变量Block

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