OC对象 - Block 的本质

OC对象 - Block 的本质

block - 代码块,开发中常用来封装一段代码,在想要执行的时候再调用执行,也常用来做传值、事件传递等等,是不可或缺的一个特性。但是如果使用不当,容易造成循环引用等问题。那block底层到底是怎么实现的呢

1. block的本质

  • block本质其实是个OC对象,他内部也有isa指针
  • block是封装了函数调用以及函数调用环境的OC对象
  • block的底层结构是:

我们把block转成c++代码,来验证下它的底层结构

1.1 查看block底层实现

  • 写一个最基础的block
c 复制代码
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        void(^block)(void) = ^{
            printf("block~~~\n");
        };
        block();
    }
    return 0;
}
  • 转成c++代码

  • struct __block_impl impl对应的就是 block 的底层结构

1.1.1 初始化block

  • 刚刚声明的 block
scss 复制代码
void(^block)(void) = ^{
            printf("block~~~\n");
        };

实际上它转成了

scss 复制代码
void(*block)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));

他是初始化了一个__main_block_impl_0结构体,

__main_block_impl_0构造方法接收两个参数void *fpstruct __main_block_desc_0 *desc,可以看到此处代码传入的值是(void *)__main_block_func_0&__main_block_desc_0_DATA

这两个参数传进__main_block_impl_0后赋值给了 impl.FuncPtrDesc

  • (void *)__main_block_func_0对应的是:

其实就是把我们 block 里面所写的代码封装成函数传进去。

Desc则是block的描述信息,对应结构体__main_block_desc_0 reserved仅是个保留字段,Block_size则表示 block 的大小

1.1.2 调用block

我们初始化block后,通过block()调用了block

底层对应转成这样:

scss 复制代码
((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);

他实际上就是帮我们调用了刚刚初始化时传入的保存在impl.FuncPtr的函数

此时block里面的代码就被调用执行了

@oubijiexi

相关推荐
倔强的石头_4 分钟前
【Linux指南】动静态库与链接机制:从原理到实践
linux
赏点剩饭77828 分钟前
linux中的hostpath卷、nfs卷以及静态持久卷的区别
linux·运维·服务器
神鸟云36 分钟前
DELL服务器 R系列 IPMI的配置
linux·运维·服务器·网络·边缘计算·pcdn
herderl1 小时前
**僵尸进程(Zombie Process)** 和**孤儿进程(Orphan Process)**
linux·运维·服务器·网络·网络协议
lepton_yang1 小时前
Zephyr下控制ESP32S3的GPIO口
linux·嵌入式硬件·esp32·zephyr
泽02021 小时前
Linux 编译器 gcc 与 g++
linux·运维·服务器
G_H_S_3_2 小时前
【网络运维】Playbook项目实战:基于 Ansible Playbook 一键部署 LNMP 架构服务器
linux·运维·服务器·网络·ansible
yuxb732 小时前
Ansible 学习笔记:变量事实管理、任务控制与文件部署
linux·运维·笔记
岚天start3 小时前
Linux sar命令详细使用指南
linux·运维·服务器·负载·sar·磁盘io·sysstat