iOS:ARC内存管理知识总结(一)

1. ARC的基本概念

  • 在objc中采用automatic reference counting 机制, 让编译器来进行内存管理。在降低程序崩溃,内存管理泄漏等风险的同时,很大程度减少了程序员的工作量。
    -------摘自苹果发开者文档

  • 自动引用计数的思维方式:

  • 自己生成的对象,自己持有。

  • 非自己生成的对象,自己也能持有。

  • 再需要自己持有对象时释放。

  • 非自己持有的对象无法释放。

  • 对象操作与OBJC方法的对应。

  • 生成并持有对象 ------ alloc/new/copy/mutableCopy(这些方法意味着自己生成的对象只能自己持有)

  • 持有对象 ------ retain

  • 释放对象 ------ release

  • 销毁对象 ------ dealoc

  • 非自己生成的对象,自己也能持有

/*取得的对象存在,但自己不持有对象*/
       id obj = [NSMutableArray array]//取得非自己生成的对象
       [obj retain]//自己持有对象
  • 对几种内存情况的总结
    • alloc/new/copy/mutableCopy这些方法自己生成且持有。
  • 如果是array等方法生成的对象不是自己持有的,但是可以通过调用retain方法变成自己持有。
  • 如果对自己生成且持有的对象使用```autorelease```方法,可以是对象存在但自己不持有。如:
    
id obj = [[obj alloc] init];
      [obj autorealease];
/*取得对象存在,但自己不持有*/
-  objc中,```array```等生成对象但自己不持有的方法内部是通过```autorealease```方法实现的。

2.alloc/retain/release/dealloc方法的实现

  • alloc方法的内部实现:
    alloc方法内部调用的是allocWithZone:(NSDefaultZone())这个对象方法。该方法又调用了NSAllocateObject()函数。所以在这里只需关注该方法即可。
    该方法内部如下:
struct obj_layout
{
    NSInteger retained;
};
inline id 
NSAllocateObject(Class aClass,NSInteger extraBytes,NSZone *zone){
    int size = 计算容纳对象所需内存大小;
    id new = NSZoneMalloc(zone,size);
    memset (new,0,size);
    new = (id) & ((struct obj_layout *) new)[1];

}
  • 该函数将该内存空间置为0(包括 obj_layout这个结构体),且返回一个作为对象而使用的指针。

该内存空间内部,也就是对象所处的内存空间内部,头部是struct obj_layout这个结构体,接下来才是对象。alloc方法返回的指针指向的头部以下的内容。也就是除去结构体的内容。

可以通过查看retainCount这个方法的源代码验证以上结论:

- (NSUInteger)retainCout{
    return NSExtraRefcount(self)+1;//需要注意此处的+1,对象内存的头部结构体里的retain其实为0,所以+1。
}

inline NSUInteger 
NSExtraRefcount(id anObject){
    return((struct obj_layout *) anObject)[-1].retained;//此处需要注意-1,因为要访问的实质对象所处内存空间的头部,也就是结构体obj_layout.
}

3.retain,release即delloc方法的实现。

无论是retain还是release,其实内部都是通过操作对象所处内存空间头部的结构体来实现的。

  • retain:
((struct obj_layout *) anObject)[-1].retained++;
  • release:这个方法的内部实现稍微要比retain复杂一点,是因为要判断引用计数是否为0,如果为0 则需要销毁对象(即调用dealloc方法)
- (void)release{
    if(NSDecrementExtraRefCountWasZero(self)){
        [self dealloc];
    }
}
BOOL
NSDecrementExtraRefCountWasZero(id anObject){
    if (obj_layout结构体内部的retained是否为0)
    {
        若为0,返回yes
    }else{
        返回no
    }
}
  • delloc方法:调用NSDellocatteObject()函数,其内部freestruct objc_layout结构体。
    以上即是alloc,release,retain,delloc方法在GNUstep的实现,*总结如下:
    1.在Objective-C的对象中存在引用计数这一数值;
    2.调用alloc,retain方法后,引用计数+1;
    3.引用计数为0时,调用delloc方法销毁对象。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容