- block本质上也是一个OC对象,因为它的内部也有个isa指针
- block是封装了函数调用以及函数调用环境的OC对象
int main(int argc, const char * argv[]) { @autoreleasepool { int age = 10; int height = 10; //声明block变量 void(^myBlock)(void) = ^{ NSLog(@"age is %d height is %d",age,height); }; //调用block myBlock(); } return 0; }
通过clang编译器执行编译成C++代码:
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m
// 一: block底层数据结构 struct __main_block_impl_0 { struct __block_impl impl; // 1: impl 结构体 struct __main_block_desc_0* Desc; // 2: block描述信息的结构体 int age; //3:捕获的外部变量 int height; //4: 和结构体同名的构造函数 ( C++语法 , 类似于 OC 的init方法,返回一个结构体对象,类似于返回self) __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _age, int _height, int flags=0) : age(_age), height(_height) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; Desc = desc; } };
//struct __block_impl 结构体 struct __block_impl { void *isa; //指向 block 的类型 int Flags;//按位表示block的附加信息 int Reserved;//保留变量 void *FuncPtr; //封装了执行 block 代码块的函数地址 };
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函数 int main(int argc, const char * argv[]) { /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; int age = 10; int height = 10; //声明block void(*myBlock)(void) = &__main_block_impl_0(//调用block数据结构中的同名的构造函 __main_block_func_0,//封装了block要执行的代码块的函数 &__main_block_desc_0_DATA,//block描述信息的结构体 age,//把局部变量当做参数传入 height ); //调用block myBlock->FuncPtr(myBlock); } return 0; }
// block底层数据结构 struct __main_block_impl_0 { struct __block_impl impl; // 1: impl 结构体 struct __main_block_desc_0* Desc; // 2: block描述信息的结构体 int age;//3: 捕获的外部变量 int height; //4: 同名的构造函数 ( C++语法 ) __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _age, int _height, int flags=0) : age(_age), height(_height) { impl.isa = &_NSConcreteStackBlock; //isa 指向了block的类型 impl.Flags = flags;//按位表示block的一些附加信息,这里是0 impl.FuncPtr = fp; // 封装了block要执行的代码块的函数 Desc = desc; //block描述信息的结构体 } };
总结:
- block的底层就是一个
struct _block_impl_0
类型的结构体,这个结构体中包含一个isa
指针,所以从更深的层次来说,block就是一个OC对象. - block结构体中又包含
impl
和Desc
结构体,impl
结构体中有一个非常重要的成员FuncPtr
,FuncPtr
是一个指针,指向了封装了blcok代码块的函数,我们看到调用block的代码:myBlock()
的底层实际上是这样子:myBlock->FuncPtr(myBlock);
,可以看出调用block其实就是调用了FuncPtr()这个函数 Desc
结构体存放了block的大小