内存管理:autorelease、autoreleasepool

一、Autoreleasepool

自动释放池,统一管理内部的局部变量。
autorelease就是将对象放入到对应的autoreleasepool中,当autoreleasepool作用域解决的时候,会将内部的变量全部进行一次release操作

二、main函数中具有一个全局的autoreleasepool

main函数中我们的主程序入口UIApplicationMain是被包裹再autoreleasepool内部的,而UIApplicationMain会开启RunLoop导致程序正常情况下会持续运行,不会退出,也就是autoreleasepool的作用域一直存在。那么里面的对象是如何被销毁的呢

三、源码分析

int main(int argc, char * argv[]) {
    @autoreleasepool {
//UIApplicationMain 内部开启的引用循环
        return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
    } // 循环不结束,不会进入次处,autoreleasepool 不会被释放才对
} 

在命令行中编译main.m文件

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m

通过上面的命令会得到一个C++的文件

int main(int argc, char * argv[]) {
    NSString * appDelegateClassName;
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        appDelegateClassName = NSStringFromClass(((Class (*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("AppDelegate"), sel_registerName("class")));
    }
    return UIApplicationMain(argc, argv, __null, appDelegateClassName);
}

通过源码发现
之前OC中的@autoreleasepool {}变成了

{
__AtAutoreleasePool __autoreleasepool; 
......
}

__AtAutoreleasePool的内部结构是

struct __AtAutoreleasePool {
// 这个结构内部只有两个函数,一个是构造函数,一个是析构函数
 __AtAutoreleasePool() {// 构造函数,创建的时候调用,内部调用了一个push的操作
atautoreleasepoolobj = objc_autoreleasePoolPush();
}
  ~__AtAutoreleasePool(){// 析构函数,销毁的时候调用,内部调用了一个pop操作
objc_autoreleasePoolPop(atautoreleasepoolobj);
}
  void * atautoreleasepoolobj;
};

//objc_autoreleasePoolPush push操作内部调用了一个叫AutoreleasePoolPage的push方法
void *
objc_autoreleasePoolPush(void)
{
    return AutoreleasePoolPage::push();
}
// pop操作里调用了 AutoreleasePoolPage 的pop方法
void
objc_autoreleasePoolPop(void *ctxt)
{
    AutoreleasePoolPage::pop(ctxt);
}

由此可见,最终管理对象的是通过 AutoreleasePoolPage

class AutoreleasePoolPage 
{
    static pthread_key_t const key = AUTORELEASE_POOL_KEY;
    static uint8_t const SCRIBBLE = 0xA3;  
// AutoreleasePoolPage固定大小为4096个字节
    static size_t const SIZE = PAGE_MAX_SIZE;//PAGE_MAX_SIZE = 4096
    static size_t const COUNT = SIZE / sizeof(id);

    magic_t const magic;
    id *next;//指针指向最后一个对象的地址
    pthread_t const thread;// 线程
    AutoreleasePoolPage * const parent;//上一个AutoreleasePoolPage
    AutoreleasePoolPage *child;// 下一个AutoreleasePoolPage
    uint32_t const depth;
    uint32_t hiwat;
....
}

1.每一个AutoreleasePoolPage对象占用4096个字节,除了用来存储它内部的变量 ,剩余的空间是用来存储对象的地址
2.每一个AutoreleasePoolPage能够存储的对象数量是有限的
3.所有的AutoreleasePoolPage是都通双向链表的形式连接到一起,当前AutoreleasePoolPage无法继续存储的时候,会创建的AutoreleasePoolPage继续存储

id * begin() {//用来计算存储对象地址的开始位置
// 自身内存地址的起始位置加上本身占用内存的大小
        return (id *) ((uint8_t *)this+sizeof(*this));
    }

    id * end() {// 返回存储对象结束的位置
// 自身内存地址起始位置+ SIZE ,size是一个固定值4096
        return (id *) ((uint8_t *)this+SIZE);
    }

AutoreleasePoolPage push内部逻辑

static inline void *push() 
    {
        id *dest;
        if (DebugPoolAllocation) {
            // 创建一个新的AutoreleasePoolPage
            dest = autoreleaseNewPage(POOL_BOUNDARY);
        } else {
 // 创建一个新的AutoreleasePoolPage
            dest = autoreleaseFast(POOL_BOUNDARY);
        }
        assert(dest == EMPTY_POOL_PLACEHOLDER || *dest == POOL_BOUNDARY);
        return dest;
    }

 // 创建一个新的AutoreleasePoolPage
static inline id *autoreleaseFast(id obj)
    {
        // 获取当前能够添加对象的page
        AutoreleasePoolPage *page = hotPage();
        if (page && !page->full()) {//如果对象没有满,向里面添加
            // push的时候传入的是一个POOL_BOUNDARY,说明push的时候会做一个标记点
            return page->add(obj);
        } else if (page) {
            // 当前的page已经满了,创建新的page
            return autoreleaseFullPage(obj, page);
        } else {
            return autoreleaseNoPage(obj);
        }
    }
//如果page满了,创建新的page继续添加
static __attribute__((noinline))
    id *autoreleaseFullPage(id obj, AutoreleasePoolPage *page)
   {
        assert(page == hotPage());
        assert(page->full()  ||  DebugPoolAllocation);

        // 循环遍历page,当page的child没有值的时候,新建一个page
        do {
            if (page->child) page = page->child;
            else page = new AutoreleasePoolPage(page);
        } while (page->full());
        // 并且将这个新的page作为hotPage,向内部添加一个obj
        // obj有可能是一个Pool_BOUNDARY边界,有可能是对象的地址
        setHotPage(page);
        return page->add(obj);
    }
// add的添加方式
id *add(id obj)
    {
        assert(!full());
        unprotect();
        id *ret = next;  
        *next++ = obj;// 将属性next 指向当前添加的对象的地址
        protect();
        return ret;
    }

当我们@autoreleasepool 的时候,首先做的就是push,进栈的操作,将一个POOL_BOUNDARY做为一个边界值放到第一个位置,每一次@autoreleasepool都会产生一个POOL_BOUNDARY,AutoreleasePoolPage能够存储的数量有限,当page无法继续存储的时候,会创建新的AutoreleasePoolPage,并将page 的child指向它

AutoreleasePoolPage pop的逻辑

void
objc_autoreleasePoolPop(void *ctxt)
{
 // 向pop函数中传递的是当前__autoreleasepool的地址
    AutoreleasePoolPage::pop(ctxt);
}
static inline void pop(void *token) 
    {
        AutoreleasePoolPage *page;
        id *stop;

        // 判断token不是不是空的pool占位符
        if (token == (void*)EMPTY_POOL_PLACEHOLDER) {
            // 如果是顶级的占位符.
            if (hotPage()) {// 判断有没有AutoreleasePoolPage
               // 获取开始边界值的位置
                pop(coldPage()->begin());
            } else {
                // AutoreleasePoolPage 没有被使用过,清空
                setHotPage(nil);
            }
            return;
        }

        // 通过token地址获取AutoreleasePoolPage
        page = pageForPointer(token);
        stop = (id *)token;
        // 如果stop的地址 不等于POOL_BOUNDARY的地址,
        if (*stop != POOL_BOUNDARY) {
            // 赋值stop 等于begin()位置的值,POOL_BOUNDARY,判断是否有值
            // 判断page 的parent是否有值
            //
            if (stop == page->begin()  &&  !page->parent) {
            } else {
                return badPop(token);
            }
        }

        if (PrintPoolHiwat) printHiwat();

        // 从next位置一直到stop位置,中间的对象都进行一次relase操作
        page->releaseUntil(stop);

        // 如果page为空了
        if (DebugPoolAllocation  &&  page->empty()) {
            // 获取它的parent 的page,销毁当前的page
            // 将parent中存储的page作为当前的page
            AutoreleasePoolPage *parent = page->parent;
            page->kill();
            setHotPage(parent);
            // 如果page 为空,parent为空,那么销毁page
        } else if (DebugMissingPools  &&  page->empty()  &&  !page->parent) {
            // special case: delete everything for pop(top) 
            // when debugging missing autorelease pools
            page->kill();
            setHotPage(nil);
        } 
        else if (page->child) {
            // hysteresis: keep one empty child if page is more than half full
            if (page->lessThanHalfFull()) {
                page->child->kill();
            }
            else if (page->child->child) {
                page->child->child->kill();
            }
        }
    }

调用pop的时候,会根据POOL_BOUNDARY边界点的位置和next位置进行release操作。
手动创建的@autoreleasepool,会在 开的时候和结束的时候分别调用 AutoreleasePoolPage的push 和pop
那么,main函数里面的那个@autoreleasepool是何时进行 释放对象的呢?

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,490评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,581评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,830评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,957评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,974评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,754评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,464评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,357评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,847评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,995评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,137评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,819评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,482评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,023评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,149评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,409评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,086评论 2 355