iOS - block - C++基础分析

[toc]

参考

block - C++基础分析

获取 C++源码

在 main.m 中, 用 OC 实现一个简单的 block

#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        void (^myBlock)(void) = ^() {
            NSLog(@"hello block");
        };
        myBlock();
    }
    return 0;
}

生成C++代码

分析 C++ 源码

相关结构体 / 函数: ★★

__block_impl

定义在 main.cpp 最上面, block 公用结构体, 所有 block 都使用该结构体;

是结构体 __main_block_impl_0 的第1个成员, 也是其核心成员, 封装了 block 的实现;

// block的父类结构体
struct __block_impl {
    void *isa; // 指向block的具体类型  // 有isa指针, 就是属于对象范畴
    int Flags;
    int Reserved;
    void *FuncPtr; // 函数指针 FunctionPointer, 指向block的实现 __funcName_block_func_0
};
__main_block_impl_0

block变量的实际类型;

  • 该结构体命名规律:

    对于局部block: __funcName_block_impl_0; 其中, funcName是block所在函数的name; _0表示是该函数里第0个block。

    对于全局block: __blockName_block_impl_0

struct __main_block_impl_0 {
    struct __block_impl impl; // 封装了函数实现的结构体
    struct __main_block_desc_0 *Desc;  // 里面有内存管理函数, Block_size表示block的大小

    // 该结构体的构造函数
    // 这就是<定义block>时, 调用的初始化方法, 这个结构体的地址最后赋值给了myBlock, 即定义block, 所以说block的本质是结构体
    // *fp  函数实现的函数指针; 调用时入参需要传入一个地址, 即 (void *)__main_block_func_0
    // *desc  占用大小的描述指针; 调用时入参需要传入一个地址, 即 &__main_block_desc_0_DATA
    // flags  默认是0
    __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int flags=0) {
        impl.isa = &_NSConcreteStackBlock; // 注意不是Global, 见下面分析
        impl.Flags = flags;
        impl.FuncPtr = fp; // 见下面分析
        Desc = desc;
    }
};
关于 impl.isa ★★

clang命令转换, 未捕获变量的局部block, 其 isa 指向 _NSConcreteStackBlock, 说明, 普通局部block 本质上是 __NSStackBlock__ 类的实例;

但打印block对象, 却输出__NSGlobalBlock__, 猜测是因为没有捕获变量, 编译器实际上就将普通block当做 __NSGlobalBlock__ 类来处理了 (参考<BC - 存储域 C>)。

impl.FuncPtr = fp;

结构体构造函数中的函数指针赋值, fp 被赋值给 __main_block_impl_0 的成员变量 impl 的成员变量 FuncPtr;

fp 是函数入参, 是个函数指针, 传入的就是block初始化时传入的 (void *)__main_block_func_0 , 也就是封装了block代码块的函数。

所以, 在<调用block>时, 可以找到这个函数指针并调用;


__main_block_desc_0

block 的描述struct;

__main_block_impl_0 的第2个成员;

其实例 __main_block_desc_0_DATA 是构造函数的第2个入参;

  • 该结构体命名规律:

局部block: __funcName_block_desc_0; 命名规则同impl。

全局block: __blockName_desc_0

static struct __main_block_desc_0 {
    size_t reserved; // 预留参数
    size_t Block_size; // block结构体占用的内存大小
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0) };
__main_block_func_0

block 中的执行代码被封装成该函数。

__main_block_impl_0 结构体构造函数的第1个入参 void *fp ;

__main_block_impl_0 初始化时, 作为函数指针 fp 传给 __main_block_impl_0impl.FuncPtr

  • 该函数命名规律:

局部block: __funcName_block_func_0; 命名规则同impl

全局block: __blockName_func_0

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
    NSLog((NSString *)&__NSConstantStringImpl__var_folders_gc_5fkhcz0n6px48vzc744hmp6c0000gn_T_main_eef954_mi_0);
}

函数入参 __cself 是指向 __main_block_impl_0 类型的结构体的指针, 类似 OC 的 self, 可以通过 __cself->val 对变量进行访问。
也就是说: 调用时, 会将 block 指针传入 __main_block_func_0 函数中, 便于取出 block 捕获的值。

main() 分析

// main.cpp
int main(int argc, char * argv[]) { 
    { __AtAutoreleasePool __autoreleasepool; /* @autoreleasepool */  // 自动释放池
        void (*myBlock)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA)); // <定义block>
        ((void (*)(__block_impl *))((__block_impl *)myBlock)->FuncPtr)((__block_impl *)myBlock); // <调用block>
    }
    return 0;
}

自动释放池
苹果通过声明一个 结构体类型 __AtAutoreleasePool 的局部变量 __autoreleasepool 实现了 @autoreleasepool{}

__AtAutoreleasePool __autoreleasepool;

定义block ★

block的结构体 __main_block_impl_0 初始化

void (*myBlock)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));
拆解分析:

① 这句 __main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA) , 是结构体 __main_block_impl_0 的初始化, 生成一个结构体变量;

② 然后用 "&" 对生成的结构体变量取址, 得到结构体指针;

③ 最后强转成 (void (*)()) 类型, 赋值给 myblock

结论 --- 定义 block 的步骤:

① 首先创建一个函数 __main_block_func_0, 该函数封装了 block 要执行的代码;

② 函数 __main_block_func_0 的指针作为结构体 __main_block_impl_0 初始化的入参传入; (因此后面可以从结构体中取出函数指针进行调用。)

③ 把该结构体的地址赋值给myBlock。 (也就是说: block 指向的是结构体__main_block_impl_0 的地址, 即: block 实质是 __main_block_impl_0 类型的变量。)


调用block ★

找到并调用函数指针 FuncPtr

((void (*)(__block_impl *))((__block_impl *)myBlock)->FuncPtr)((__block_impl *)myBlock);
拆解分析:

(void (*)(__block_impl *)) 是函数类型(就是 __main_block_func_0 的类型), 该类型的函数, 入参为 __block_impl * 类型, 返回值为 void 类型;

((__block_impl *)myBlock) 是将 myBlock 强转成 __block_impl 类型, 并找到结构体 __block_impl 的成员 FuncPtr;

类型强转过程:

在 <定义block> 中得知: block 实质是 __main_block_impl_0 类型的变量;

结构体 __main_block_impl_0 的第1个成员是结构体 __block_impl ,

所以 __main_block_impl_0 的首地址就是 __block_impl 的内存地址, 故而可以强转成功,

并可以通过操作符 "->" 找到其结构成员 FunPtr (函数指针)。

((void (*)(__block_impl *))((__block_impl *)myBlock)->FuncPtr) 是把函数指针 FuncPtr 强转为(void (*)(__block_impl *)) 类型;

FuncPtr 指向封装了block所要执行代码的函数地址 __main_block_func_0 (在__main_block_impl_0 初始化时传入的), 所以调用此函数, 就会执行代码块中的代码。

⑤ 调用函数 __main_block_func_0 需要传入 struct __main_block_impl_0 * 类型的参数, 所以 block本身 ((__block_impl *)myBlock) 便作为函数入参传入, 以便函数内部使用block捕获到的变量;


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,258评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,335评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,225评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,126评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,140评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,098评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,018评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,857评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,298评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,518评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,678评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,400评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,993评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,638评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,801评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,661评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,558评论 2 352

推荐阅读更多精彩内容