block内部结构

让我们写一个block

void exampleBlock() {
// NSConcreteStackBlock
int a = 1;
__block int b = 2;
int(^blockTest0)(int c) = ^(int c){
return a + b + c;
};
int c = 3;
blockTest0(c); // NSConcreteGlobalBlock
void(^blockTest2)(void) = ^(void){
;
};
blockTest2();
}

用clang转成c分析下

clang -rewrite-objc block.c

能够看到他们的定义是

struct __exampleBlock_block_impl_0 {
struct __block_impl impl;
struct __exampleBlock_block_desc_0* Desc;
int a;
__Block_byref_b_0 *b; // by ref
__exampleBlock_block_impl_0(void *fp, struct __exampleBlock_block_desc_0 *desc, int _a, __Block_byref_b_0 *_b, int flags=0) : a(_a), b(_b->__forwarding) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
}; // __block_impl
struct __block_impl {
void *isa;
int Flags;
int Reserved;
void *FuncPtr;
}; // __exampleBlock_block_desc_0
struct __exampleBlock_block_impl_0 {
struct __block_impl impl;
struct __exampleBlock_block_desc_0* Desc;
int a;
__Block_byref_b_0 *b; // by ref
__exampleBlock_block_impl_0(void *fp, struct __exampleBlock_block_desc_0 *desc, int _a, __Block_byref_b_0 *_b, int flags=0) : a(_a), b(_b->__forwarding) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
}; // __exampleBlock_block_impl_2
struct __exampleBlock_block_impl_2 {
struct __block_impl impl;
struct __exampleBlock_block_desc_2* Desc;
__exampleBlock_block_impl_2(void *fp, struct __exampleBlock_block_desc_2 *desc, int flags=0) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
};

初始化和运行代码

void exampleBlock() {
// blockTest0
int a = 1;
__attribute__((__blocks__(byref))) __Block_byref_b_0 b = {(void*)0,(__Block_byref_b_0 *)&b, 0, sizeof(__Block_byref_b_0), 2}; int(*blockTest0)(int c) = (int (*)(int))&__exampleBlock_block_impl_0((void *)__exampleBlock_block_func_0, &__exampleBlock_block_desc_0_DATA, a, (__Block_byref_b_0 *)&b, 570425344); int c = 3;
((int (*)(__block_impl *, int))((__block_impl *)blockTest0)->FuncPtr)((__block_impl *)blockTest0, c); // blockTest2
void(*blockTest2)(void) = (void (*)())&__exampleBlock_block_impl_2((void *)__exampleBlock_block_func_2, &__exampleBlock_block_desc_2_DATA);
((void (*)(__block_impl *))((__block_impl *)blockTest2)->FuncPtr)((__block_impl *)blockTest2);
}

我们先看看blockTest2,它是由 结构体impl, 结构体Desc, 构造方法__exampleBlock_block_impl_2() 组成展开后是

  • *isa 指向该实例对象(代码里是NSConcreteStackBlock,事实上应该是NSConcreteGlobalBlock)
  • Flags 用于按bit位表示一些block的附加信息
  • reserved 保留变量
  • *FuncPtr 函数指针,指向详细的block实现的函数调用地址(代码里是__exampleBlock_block_func_2)
static void __exampleBlock_block_func_2(struct __exampleBlock_block_impl_2 *__cself) {
;
}
  • size_t reserved 这个传进来的是0
  • Block_size 结构体的大小
static struct __exampleBlock_block_desc_2 {
size_t reserved;
size_t Block_size;
} __exampleBlock_block_desc_2_DATA = { 0, sizeof(struct __exampleBlock_block_impl_2)};

然后我们在看blockTest,它比blockTest2多了2个变量a, b

  • int a; 外部变量a,

  • Block_byref_b_0 *b; 加了block修饰的b, 传的是 Block_byref_b_0

在生成的初始化代码中则多了3个传入值

    int(*blockTest0)(int c) = (int (*)(int))&__exampleBlock_block_impl_0((void *)__exampleBlock_block_func_0, &__exampleBlock_block_desc_0_DATA, a, (__Block_byref_b_0 *)&b, 570425344);
  • a是这个是直接传值进去,然后被复制给 a

  • b是传的地址, 是把 __Block_byref_b_0 赋值给 b

__Block_byref_b_0这个结构体是

struct __Block_byref_b_0 {
void *__isa;
__Block_byref_b_0 *__forwarding;
int __flags;
int __size;
int b;
};

__forwarding 是一个指向自己的指针.

__Block_byref_b_0 的初始化代码例如以下:

__attribute__((__blocks__(byref))) __Block_byref_b_0 b = {(void*)0,(__Block_byref_b_0 *)&b, 0, sizeof(__Block_byref_b_0), 2};

我们能够看出a是直接复制进去的,b是被转到了一个结构体里,然后吧这个结构体的指针传进去,所以block不能改动a,能改动b.

  • 570425344 这个应该是传给Flags

blockTest0的Desc和blockTest2也有所不同,展开后是

static struct __exampleBlock_block_desc_0 {
size_t reserved;
size_t Block_size;
void (*copy)(struct __exampleBlock_block_impl_0*, struct __exampleBlock_block_impl_0*);
void (*dispose)(struct __exampleBlock_block_impl_0*);
} __exampleBlock_block_desc_0_DATA = { 0, sizeof(struct __exampleBlock_block_impl_0), __exampleBlock_block_copy_0, __exampleBlock_block_dispose_0};

多了2个函数指针copy, dispose,对于在调用前后改动对应变量的引用计数, 分别指向

static void __exampleBlock_block_copy_0(struct __exampleBlock_block_impl_0*dst, struct __exampleBlock_block_impl_0*src) {_Block_object_assign((void*)&dst->b, (void*)src->b, 8/*BLOCK_FIELD_IS_BYREF*/);}

static void __exampleBlock_block_dispose_0(struct __exampleBlock_block_impl_0*src) {_Block_object_dispose((void*)src->b, 8/*BLOCK_FIELD_IS_BYREF*/);}

再来看下blockTest0的*FuncPtr

 static int __exampleBlock_block_func_0(struct __exampleBlock_block_impl_0 *__cself, int c) {
__Block_byref_b_0 *b = __cself->b; // bound by ref
int a = __cself->a; // bound by copy return a + (b->__forwarding->b) + c;
}

如可以看到的a使用输入的副本a, b使用一种结构,其中b

版权声明:本文博主原创文章。博客,未经同意不得转载。

最新文章

  1. RESTful 接口调试分享利器 restc
  2. 【转】COM技术内幕(笔记)
  3. 一些Python的惯用法和小技巧:Pythonic
  4. LightOJ1018 Brush (IV)(状压DP)
  5. .NET常用方法收藏
  6. hdu 3342 Legal or Not
  7. GDB调试方法精粹
  8. Help View修复
  9. mac上做透明图片, png, alpha
  10. PS切图篇(一)---界面设置
  11. ACM Meteor Shower
  12. Copula函数
  13. 【转】关于 python ImportError: No module named 的问题
  14. 洛谷SP22343 NORMA2 - Norma(分治,前缀和)
  15. Oracle 11g R2 for Win10(64位)的安装步骤
  16. ubuntu 关闭 笔记本键盘背景灯
  17. Spring Security教程(三):自定义表结构
  18. 在Asp.net WebAPI使用Session
  19. hbase shell 启动报错
  20. C#中null值属于什么变量类型

热门文章

  1. error app/styles/components/iconfont.scss (Line 12: Invalid GBK character "\xE5")
  2. NIO 入门(转)
  3. 利用动态图层实现数据的实时显示(arcEngine IDynamiclayer)
  4. 【重拾Effective Java】一
  5. python的报错
  6. Satisfying memory ordering requirements between partial reads and non-snoop accesses
  7. 在RedHa上安装MRTG监控网卡流量
  8. js进阶正则表达式7点数字字母空格(w d s)(小写表原意,大写表反义)(特殊字符要加反斜杠:var reg22=/\W/g)
  9. net的微服务架构
  10. Tomcat8及之后版本出现的The valid characters are defined in RFC 7230 and RFC 3986