哪些对象会加入自动释放池呢?

有没有好奇过, 自动释放池会延迟释放对象,那到底哪些对象会加入制动释放池呢?

先科普一下main函数里的结构

以前的main函数的结构



现在的main函数结构



发现有啥不同了不?
区别就在于

return UIApplicationMain(argc, argv, nil, appDelegateClassName);
老项目结构放在了@autoreleasepool {}里面

为啥新项目结构后来苹果选择放在@autoreleasepool {}外面了呢?
比如如下情况,这个o对象就无法释放了

int main(int argc, char * argv[]) {
    @autoreleasepool {
        objc *o = [objc create]; /// 假如该对象会加入自动释放池,此时o对象将永远无法释放了
        return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
    }
}

回归正题

  • 方法1
    viewDidLoad弱持有对象,viewWillAppear访问
    由于这两个方法是在同一个runloop循环中,如果加入了自动释放池,在viewWillAppear还是可以访问到的。
    注(进入runloop会生成一个自动释放池,退出一个runloop循环会释放自动释放池及释放池里的对象)

上代码

@interface ViewController ()
@property (weak, nonatomic) NSMutableArray *arr;
@end

@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    NSMutableArray *array = [[NSMutableArray alloc] init];
    self.arr = array;
}

- (void)viewWillAppear:(BOOL)animated {
    [super viewWillAppear:animated];
    NSLog(@"self.arr   %@", self.arr);
}
  • 方法2
    终结大招
    通过_objc_autoreleasePoolPrint打印注册到自动释放池里的对象

上代码


有一个对象加入了自动释放池(2 releases pending, 其中有个哨兵对象)

怀疑不同环境下,结果也会不一样,同样的代码,换个环境,有两个数组加入了自动释放池


网上很多解释
什么objc_autoreleaseReturnValue,然后objc_retainAutoreleasedReturnValue(https://www.jianshu.com/p/5ec8fdef8e7e)的就不会加入自动释放池(自行百度google),但还是得自己验证,每个版本都有可能有变化,不可尽信书(况且网上的文章还不是书),也不可不看书。

现在来分析下_objc_autoreleasePoolPrint这个是否有可信度呢?

上源码

void 
_objc_autoreleasePoolPrint(void)
{
    AutoreleasePoolPage::printAll();
}

进入printAll(); 关键代码,一页一页遍历

for (page = coldPage(); page; page = page->child) {
     page->print();
}

然后我们在看下每页的打印page->print();


################ POOL 0x10400a038 打印的正式占位对象

            _objc_inform("[%p]  %#16lx  %s",
                         p, (unsigned long)*p, object_getClassName(*p));

打印的正是

objc[38018]: [0x10400a040] 0x100c65300 __NSArrayM
objc[38018]: [0x10400a048] 0x100c65380 __NSArrayM

格言:不可尽信书,也不可不看书,看完此文章还需亲自己亲自验证才能得自己的理解。

你可能感兴趣的:(哪些对象会加入自动释放池呢?)