block是带有局部变量的匿名函数,在底层是被当做C语言函数在处理的。
具体分析如下
int main(int argc, const char * argv[]) {
void(^blk) (void) = ^{printf("Block\n");};
blk();
return 0;
}
上述代码通过clang -rewrite-objc 源文件名 转换成下面源代码
struct __block_imp1{
void *isa;
int Flags;
int Reserved;
void *FuncPtr;
}
struct __main_block_impl_0 {
struct __block_impl impl;
struct __main_block_desc_0* Desc;//指向__main_block_desc_0的指针
//结构体_main_block_imp1_0的构造函数,第二个参数为__main_block_desc_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;
}
};
//^{printf("Block\n");};语句转换成C语言的函数
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
printf(“Block\n")
;}
//初始换__main_block_impl_0的大小
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)
};
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;
}
(1)Block语法使用的匿名函数^{printf(“Block\n”);};实际被作为C语言函数来处理,转换后C语言函数的名字根据Block所属的函数以及在该函数中出现的顺序命名,本文命名为_main_block_fun_0(struct _main_block_imp1_0 *_cself);
_cself是指向_mian_block_imp1_0结构体的指针
(2) 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((void *)__main_block_func_0, &__main_block_desc_0_DATA));
struct __main_block_impl_0 *blk = &tmp;
代码意思是讲过Block语法生成的block赋值给block
类型的变量blk,即将__main_block_impl_0结构体的实例的指针赋值给blk。
(3) ((void ()(__block_impl ))((__block_impl )blk)->FuncPtr)((__block_impl )blk);
转化为:(*blk->impl.FuncPtr)(blk),其实就是函数指针调用函数
(4)block可以作为对象处理
因为OC中由类生成对象意味着生成由该类生成的对象的结构体实例,生成的各个对象,即由该类生成的对象的各个结构体的实例,通过isa保持该类的结构体实例指针;在OC中基于objc_class结构体的class_t结构体。
所以_NSConcreteStackBlock相当于class_t结构体实例,将block可以作为对象处理时,关于类的信息放在_NSConcreteStackBlock。
网上找到的苹果官方翻译--Block
http://download.csdn.net/detail/u011774517/9832261