AutoreleasePool

这里只是简单的串一串自己脑海里的知识。所以就简单点儿,引出今天的主题就行了。看如下代码:

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        NSLog(@"Hello, World!");
    }
    return 0;
}

很简单,很基本的main函数。经过使用命令:

clang -rewrite-objc -framework Foundation main.m -o MainClang.cpp

可以变成c++,进行查看底层的入口。编译后如下:

int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ {
        __AtAutoreleasePool __autoreleasepool;  // 这是重点

        NSLog((NSString *)&__NSConstantStringImpl__var_folders_hk_zhxz0gtj7z1djqcl5mbjdpd00000gn_T_main_76dea1_mi_0);
    }
    return 0;
}

__AtAutoreleasePool __autoreleasepool就是重点。
__AtAutoreleasePool定义如下(我也不知道该叫做类还是结构体,因为在C++里struct和其他语言中是不一样的):

struct __AtAutoreleasePool {
  __AtAutoreleasePool() {
      atautoreleasepoolobj = objc_autoreleasePoolPush();
      // 这里返回的要么是对象的储存地址,要么是一个标记EMPTY_POOL_PLACEHOLDER,用来表示当前的Page是空的
  }
  ~__AtAutoreleasePool() {
      objc_autoreleasePoolPop(atautoreleasepoolobj);
  }
  void * atautoreleasepoolobj;
};

在该结构体中,就只有三相:构造函数、析构函数和一个成员变量。

  • __AtAutoreleasePool()
    这是入口函数,可以发现在其中就做了一件事情:push

  • ~__AtAutoreleasePool()
    这个是析构函数,用于在对象释放的时候,也只干了一件事情:pop

  • objc_autoreleasePoolPush和objc_autoreleasePoolPop

void *
objc_autoreleasePoolPush(void)
{
    return AutoreleasePoolPage::push();
}

void
objc_autoreleasePoolPop(void *ctxt)
{
    AutoreleasePoolPage::pop(ctxt);
}

在这两个里面都不约而同的提到了一个类:AutoreleasePoolPage,而该类才是自动释放池的主角,可以说有了它,才有了自动释放池的一切。

  • AutoreleasePoolPage
    简单来说,是一个双向列表,它维护着一个用来管理需要自动释放的变量的栈,因为每次开辟的Page大小都是有限的,所以当当前page空间不足的时候,就会重新创建一个Page。
注意⚠️:Page是和线程相关的,如果所在线程不正确,会报错。

其主要定义如下:

class AutoreleasePoolPage 
{
    // EMPTY_POOL_PLACEHOLDER is stored in TLS when exactly one pool is 
    // pushed and it has never contained any objects. This saves memory 
    // when the top level (i.e. libdispatch) pushes and pops pools but 
    // never uses them.
#   define EMPTY_POOL_PLACEHOLDER ((id*)1)

#   define POOL_BOUNDARY nil
static pthread_key_t const key = AUTORELEASE_POOL_KEY;
    static uint8_t const SCRIBBLE = 0xA3;  // 0xA3A3A3A3 after releasing
    static size_t const SIZE = 
#if PROTECT_AUTORELEASEPOOL
        PAGE_MAX_SIZE;  // must be multiple of vm page size
#else
        PAGE_MAX_SIZE;  // size and alignment, power of 2
#endif
    static size_t const COUNT = SIZE / sizeof(id);

    magic_t const magic;
    id *next;
    pthread_t const thread;
    AutoreleasePoolPage * const parent;
    AutoreleasePoolPage *child;
    uint32_t const depth;
    uint32_t hiwat;
}
=======================PUSH开始=======================
  • push下的autoreleaseFast函数
static inline void *push() 
    {
        id *dest;
        if (DebugPoolAllocation) {
            // Each autorelease pool starts on a new pool page.
            dest = autoreleaseNewPage(POOL_BOUNDARY);
        } else {
            dest = autoreleaseFast(POOL_BOUNDARY);
        }
        assert(dest == EMPTY_POOL_PLACEHOLDER || *dest == POOL_BOUNDARY);
        return dest;
    }

可以发现其中最重要的其实就是autoreleaseFast函数:

static inline id *autoreleaseFast(id obj)
    {
        AutoreleasePoolPage *page = hotPage();
        if (page && !page->full()) {
            return page->add(obj);
        } else if (page) {
            return autoreleaseFullPage(obj, page);
        } else {
            return autoreleaseNoPage(obj);
        }
    }

hotPage个人理解就是当前正在使用的Page,该Page可能还是空,没有创建,和标记EMPTY_POOL_PLACEHOLDER做比较,然后返回空;如果创建了,则需要检查一下是否符合一些要求(这之后唯一能看懂的就是线程要能对应,如果不正确,会出错):

static inline AutoreleasePoolPage *hotPage() 
    {
        AutoreleasePoolPage *result = (AutoreleasePoolPage *)
            tls_get_direct(key);    // 根据key获取到Page
        if ((id *)result == EMPTY_POOL_PLACEHOLDER) return nil;  // 检测是否还是空的,没有用过,如果是,则直接返回nil
        if (result) result->fastcheck();   // 检测是否否和一些要求,这里对当前线程也做了检测,其他的暂时没看懂
        return result;
    }

继续说回上面的函数autoreleaseFast。在这里做了如下判断:
1、当前的Page是否存在并且没有满(即空间没有使用完),如果是,则做两件事情:
  a、执行添加,并且返回当前next所在的位置给__AtAutoreleasePool的变量atautoreleasepoolobj,该位置就是当前加入pool中变量的地址,方便后期释放使用。
  b、在add方法中修改next指针位置,并把该位置上的值设置为空POOL_BOUNDARY。add方法定义如下:

id *add(id obj)  // 主要做的事情是把next移动一位,并把它的值设置为nil,然后返回
    {
        assert(!full());
        unprotect();
        // 用ret保存当前next的位置,该位置要返回给相应的pool中的变量
        id *ret = next;  // faster than `return next-1` because of aliasing
        *next++ = obj;  // 修改next指针位置,使之后移一位,并且清空当前位置上的值,++的优先级要高于*的优先级
        protect();
        return ret;
    }

2、如果存在但是当前page已经满了,调用autoreleaseFullPage函数

id *autoreleaseFullPage(id obj, AutoreleasePoolPage *page)
    {
        // The hot page is full. 
        // Step to the next non-full page, adding a new page if necessary.
        // Then add the object to that page.
        assert(page == hotPage());
        assert(page->full()  ||  DebugPoolAllocation);
        // 使用遍历,找到最后一个满了的page,然后创建一个Page
        do {
            if (page->child) page = page->child;
            else page = new AutoreleasePoolPage(page);  // 在这里完成了双向链表的链接
        } while (page->full());

        setHotPage(page);  // 设置当前page为hotPage
        return page->add(obj);  // 添加page,并且返回添加obj之前next的位置
    }

在该方法中做了如下的事情:
  a、通过循环遍历,找到最后一个page,并且调AutoreleasePoolPage初始化,完成page的链接
  b、设置新page为hotPage,即当前page
  c、执行添加操作
3、如果没有Page,则调用autoreleaseNoPage添加

id *autoreleaseNoPage(id obj)
    {
        // "No page" could mean no pool has been pushed
        // or an empty placeholder pool has been pushed and has no contents yet
        assert(!hotPage());

        bool pushExtraBoundary = false;
        if (haveEmptyPoolPlaceholder()) {
            // We are pushing a second pool over the empty placeholder pool
            // or pushing the first object into the empty placeholder pool.
            // Before doing that, push a pool boundary on behalf of the pool 
            // that is currently represented by the empty placeholder.
            pushExtraBoundary = true;
        }
        else if (obj != POOL_BOUNDARY  &&  DebugMissingPools) {
            // We are pushing an object with no pool in place, 
            // and no-pool debugging was requested by environment.
            _objc_inform("MISSING POOLS: (%p) Object %p of class %s "
                         "autoreleased with no pool in place - "
                         "just leaking - break on "
                         "objc_autoreleaseNoPool() to debug", 
                         pthread_self(), (void*)obj, object_getClassName(obj));
            objc_autoreleaseNoPool(obj);
            return nil;
        }
        else if (obj == POOL_BOUNDARY  &&  !DebugPoolAllocation) {
            // We are pushing a pool with no pool in place,
            // and alloc-per-pool debugging was not requested.
            // Install and return the empty pool placeholder.
            return setEmptyPoolPlaceholder();
        }

        // We are pushing an object or a non-placeholder'd pool.

        // Install the first page.
        AutoreleasePoolPage *page = new AutoreleasePoolPage(nil);  // 因为当前自动释放池中还没有page,所以这里创建的就是第一个根page
        setHotPage(page);
        
        // Push a boundary on behalf of the previously-placeholder'd pool.
        if (pushExtraBoundary) {
            page->add(POOL_BOUNDARY);
        }
        
        // Push the requested object or pool.
        return page->add(obj);
    }
==========================PUSH完成==========================
==========================POP开始===========================

当前的pool释放的时候,会调用其析构函数进行释放,在析构函数里只调用了objc_autoreleasePoolPop函数,该函数最后调用的也是AutoreleasePoolPage下的pop函数。

  • pop
static inline void pop(void *token)  // token就是那个需要释放的对象
    {
        AutoreleasePoolPage *page;
        id *stop;

        if (token == (void*)EMPTY_POOL_PLACEHOLDER) {
            // Popping the top-level placeholder pool.
            if (hotPage()) {
                // Pool was used. Pop its contents normally.
                // Pool pages remain allocated for re-use as usual.
                pop(coldPage()->begin());   // codePage其实就是得到根Page,通过begin找到该page下第一个对象的位置
            } else {
                // Pool was never used. Clear the placeholder.
                setHotPage(nil);
            }
            return;
        }

        page = pageForPointer(token);   // 找到当前对象token所在的page
        stop = (id *)token;   // 确定停止位置,停止位置为释放的变量的地址
        if (*stop != POOL_BOUNDARY) {   // 该对象存在
            if (stop == page->begin()  &&  !page->parent) {  // 如果是个空的page,且不是根page
                // Start of coldest page may correctly not be POOL_BOUNDARY:
                // 1. top-level pool is popped, leaving the cold page in place
                // 2. an object is autoreleased with no pool
            } else {
                // Error. For bincompat purposes this is not 
                // fatal in executables built with old SDKs.
                return badPop(token);
            }
        }

        if (PrintPoolHiwat) printHiwat();

        page->releaseUntil(stop);  // 释放当前page上的变量

        // memory: delete empty children  释放空的page
        if (DebugPoolAllocation  &&  page->empty()) {
            // special case: delete everything during page-per-pool debugging
            AutoreleasePoolPage *parent = page->parent;
            page->kill();
            setHotPage(parent);
        } 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();
            }
        }
    }

在这里依次干了这么几件事情:

  • 判断当前的空的要释放的对象是否是空标记EMPTY_POOL_PLACEHOLDER,表示当前的Page是空的,如果是,则调用方法coldPage获取到根Page以及其内容的开始位置begin(因为每个对象内部都有自己的方法和变量,然后才是开始存储需要自动释放的变量),另外这里是迭代调用:
static inline AutoreleasePoolPage *coldPage() 
    {
        AutoreleasePoolPage *result = hotPage();
        if (result) {
            while (result->parent) {
                result = result->parent;
                result->fastcheck();
            }
        }
        return result;
    }
  • 通过方法pageForPointer获取到当前需要释放的变量所在的page的其实位置,即this的位置
static AutoreleasePoolPage *pageForPointer(const void *p) 
    {
        return pageForPointer((uintptr_t)p);
    }

    static AutoreleasePoolPage *pageForPointer(uintptr_t p) 
    {
        AutoreleasePoolPage *result;
        uintptr_t offset = p % SIZE;   // 通过取余操作,获取到当前对象在page中的偏移量

        assert(offset >= sizeof(AutoreleasePoolPage));

        result = (AutoreleasePoolPage *)(p - offset);  // 当前对象的地址减去偏移量,就是当前page的起始位置
        result->fastcheck();

        return result;
    }
  • 确定停止位置,停止位置其实就是当前需要释放的变量的地址
  • 如果停止位置为nil,即POOL_BOUNDARY,则需要确定其满足条件:hotPage的管理外部释放变量的起始位置begin,并且是根Page,否则会出错:
if (*stop != POOL_BOUNDARY) {   // 该对象存在
            if (stop == page->begin()  &&  !page->parent) {  // 如果是个空的page,且不是根page
                // Start of coldest page may correctly not be POOL_BOUNDARY:
                // 1. top-level pool is popped, leaving the cold page in place
                // 2. an object is autoreleased with no pool
            } else {
                // Error. For bincompat purposes this is not 
                // fatal in executables built with old SDKs.
                return badPop(token);
            }
        }
  • 开始释放到该位置的所有的变量(即存储位置比该位置高,或者在该位置之后的其他的page里的变量)
void releaseUntil(id *stop) 
    {
        // Not recursive: we don't want to blow out the stack 
        // if a thread accumulates a stupendous amount of garbage
        
        while (this->next != stop) {
            // Restart from hotPage() every time, in case -release 
            // autoreleased more objects
            AutoreleasePoolPage *page = hotPage();

            // fixme I think this `while` can be `if`, but I can't prove it
            while (page->empty()) {   // 找到那个有自动释放对象的page
                page = page->parent;
                setHotPage(page);
            }

            page->unprotect();
            id obj = *--page->next;  // id obj = *--(page->next)  通过将page的next指针下移一个来获取到需要释放的对象
            memset((void*)page->next, SCRIBBLE, sizeof(*page->next));
            page->protect();

            if (obj != POOL_BOUNDARY) {
                objc_release(obj);
            }
        }

        setHotPage(this);

#if DEBUG
        // we expect any children to be completely empty
        for (AutoreleasePoolPage *page = child; page; page = page->child) {
            assert(page->empty());
        }
#endif
    }
  • 调用kill()方法清除内容为空的Page(内容为空:begin == next)
    这里有一个特殊处理,如果当前page的内容少于一半的存储空间,则直接清除其空的child,否则清除其child的child。这样做可能是为了减少后续的分配空间的时间支出吧。
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完成===========================
总结

push的过程为:调用autoreleaseFast方法完成创建,在该方法中做了三个判断:

  • hotPage是否存在且没有满
    如果没有没满,则直接添加
  • hotPage是否存在
    满了,则需要调用autoreleaseFullPage,在里面进行创建新的page并链接起来
  • hotPage不存在
    调用autoreleaseNoPage,创建一个根Page

pop的过程则为:调用pop方法完成:

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

推荐阅读更多精彩内容