面试题

autorelease原理

2020-08-12  本文已影响0人  夜沐月

对象调用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对应

上一篇 下一篇

猜你喜欢

热点阅读