对象调用autorelease会在它所在的@autoreleasepool{}大括号结束的时候调用release释放,
MRC下
NSLog(@"11111");
@autoreleasepool {
CXWork *work = [[[CXWork alloc]init] autorelease];
// CXWork *work = [[CXWork alloc]init];
}
NSLog(@"22222");
进一步研究
用xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m
将main.m转成main.cpp观察
@autoreleasepool {
CXWork *work = [[[CXWork alloc]init] autorelease];
}
如下
{
__AtAutoreleasePool __autoreleasepool;//调用构造函数
CXWork *work = ((CXWork *(*)(id, SEL))(void *)objc_msgSend)((id)((CXWork *(*)(id, SEL))(void *)objc_msgSend)((id)((CXWork *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("CXWork"), sel_registerName("alloc")), sel_registerName("init")), sel_registerName("autorelease"));
// 结构体销毁,调用结构体的析构函数
}
struct __AtAutoreleasePool {
__AtAutoreleasePool() { // 构造函数 在生成结构体变量的时候调用
atautoreleasepoolobj = objc_autoreleasePoolPush();
}
~__AtAutoreleasePool() {//析构函数,在结构体销毁的时候调用
objc_autoreleasePoolPop(atautoreleasepoolobj);
}
void * atautoreleasepoolobj;
};
void *
objc_autoreleasePoolPush(void)
{
return AutoreleasePoolPage::push();
}
void
objc_autoreleasePoolPop(void *ctxt)
{
AutoreleasePoolPage::pop(ctxt);
}
自动释放池的主要底层数据结构是:__AtAutoreleasePool、AutoreleasePoolPage,调用了autorelease的对象最终都是通过AutoreleasePoolPage对象来管理的,每个AutoreleasePoolPage对象占用4096字节内存,除了用来存放它内部的成员变量,剩下的空间用来存放autorelease对象的地址,当一个AutoreleasePoolPage不够用时,就会创建多个,所有的AutoreleasePoolPage对象通过双向链表的形式连接在一起.
AutoreleasePoolPage精简版
class AutoreleasePoolPage {
magic_t const magic;
id *next;
pthread_t const thread;
AutoreleasePoolPage *const parent;
AutoreleasePoolPage * child;
unit32_t const depth;
unit32_t hiwat;
}
objc_autoreleasePoolPush()怎么把autorelease对象地址放进AutoreleasePoolPage?
调用push方法会将一个POOL_BOUNDARY入栈,并且返回其存放的内存地址
调用pop方法时传入一个POOL_BOUNDARY的内存地址,会从最后一个入栈的对象开始发送release消息,直到遇到这个POOL_BOUNDARY.(先进后出的效果).多个autoreleasePool嵌套相当于产生多个POOL_BOUNDARY来区分每个autoreleasePool的作用域存放的对象.
id *next指向了下一个能存放autorelease对象地址的区域
iOS在主线程的Runloop中注册了2个Observer
第1个Observer监听了kCFRunLoopEntry事件,会调用objc_autoreleasePoolPush()
第2个Observer监听了kCFRunLoopBeforeWaiting事件,会调用objc_autoreleasePoolPop()、objc_autoreleasePoolPush(),监听了kCFRunLoopBeforeExit事件,会调用objc_autoreleasePoolPop() 使得pop和push对应