美文网首页iOS 开发 iOS Developer
iOS:ARC内存管理知识总结(一)

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

作者: ZhengLi | 来源:发表于2016-04-03 00:46 被阅读232次

    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方法销毁对象。

    相关文章

      网友评论

        本文标题:iOS:ARC内存管理知识总结(一)

        本文链接:https://www.haomeiwen.com/subject/lyculttx.html