iOS MRC/ARC内存管理基础篇

1. 引用计数(Reference Count)

  • 也叫保留计数(retain count),表示对象被引用的次数。一个简单而有效的管理对象生命周期的方式
  • C++11中的智能指针,微软的COM,OC都是使用这个技术来实现内存管理的
  • oc中,每一个对象都会有一个记录引用次数的属性(retainCount),可以用[object valueForKey:@"retainCount"]等方式获取对象的引用计数

2. OC内存管理三个进程(针对Cocoa类,CFType不包含在内)

MRC中内存管理规则:

  • alloc ,new创建一个对象obj1,会自动让对象的引用计数为1
  • 当我们需要用一个新的指针pointer1指向上面创建的对象obj1的时候,除了赋值,还需要手动调用[obj1 retain]或者[obj1 copy]手动修改对象的引用计数加1
  • 在要超出指针pointer1的作用域的时候,我们需要让手动调用[pointer1 release],让引用计数减1

autorelease和Autoreleasepool:

Autoreleasepool理解
  • 每一个线程的 autoreleasepool 其实就是一个指针的堆栈;
  • 每一个指针代表一个需要 release 的对象或者 POOL_SENTINEL(哨兵对象,代表一个 autoreleasepool 的边界);
  • 一个 pool token 就是这个 pool 所对应的 POOL_SENTINEL 的内存地址。当这个 pool 被 pop 的时候,所有内存地址在 pool token 之后的对象都会被 release ;
  • 这个堆栈被划分成了一个以 page 为结点的双向链表。pages 会在必要的时候动态地增加或删除;
  • ARC下,我们使用@autoreleasepool{} 来使用一个AutoreleasePool,随后编译器将其改写成下面的样子,而这两个函数都是对AutoreleasePoolPage的简单封装,所以自动释放机制的核心就在于这个类
// {}中的代码
void *context = objc_autoreleasePoolPush();
objc_autoreleasePoolPop(context);
  • AutoreleasePool并没有单独的结构,而是由若干个AutoreleasePoolPage以双向链表
    的形式组合而成
  • 一个空的 AutoreleasePoolPage 的内存结构如下图所示:


    image.png

magic 用来校验 AutoreleasePoolPage 的结构是否完整;
next 指向最新添加的 autoreleased 对象的下一个位置,初始化时指向 begin() ;
thread 指向当前线程,所以AutoreleasePool是按线程一一对应的
parent 指向父结点,第一个结点的 parent 值为 nil ;
child 指向子结点,最后一个结点的 child 值为 nil ;
depth 代表深度,从 0 开始,往后递增 1;
hiwat 代表 high water mark 。
另外,当 next == begin() 时,表示 AutoreleasePoolPage 为空;当 next == end() 时,表示 AutoreleasePoolPage 已满。

ARC下的内存管理:

  • 底层依然是引用计数的东西,只不过编译器帮我们在适当的位置添加MRC中管理引用计数的代码而已
  • 编译器会在编译阶段以恰当的时间与地方给我们填上原本需要手写的retain、release、autorelease等内存管理代码,所以ARC并非运行时的特性,也不是如java中的GC运行时的垃圾回收系统;因此,我们也可以知道,ARC其实是处于编译器的特性。
  • ARC是编译器的特性,但也包含了运行期组件,所执行的优化很有意义。解释如下:原文 链接
    image.png

3. CoreFoundation的内存管理

Core Foundation 对象必须使用CFRetainCFRelease来进行内存管理。

实际上 Core Foundation 对象使用的 CFRetain 和 CFRelease 方法,可以认为与 Objective-C 对象的 retain 和 release 方法等价,所以我们可以以 MRC 的方式进行类似管理,有一个小习惯注意养成CFRelease (cfobj)之前,判断cfobj是否为nil,不为nil时再调用release

当使用Objective-C 和 Core Foundation 对象相互转换的时候,怎么处理?

必须让编译器知道,到底由谁来负责释放对象,是否交给ARC处理。只有正确的处理,才能避免内存泄漏和double free导致程序崩溃。

__bridge:只做类型转换,不
修改相关对象的引用计数,不修改所有权. 例如:原来对象是 Core Foundation ,那么对象在不用时,需要调用 CFRelease 方法。

__bridge_retained:类型转换后,将相关对象的引用计数加1

__bridge_transfer:类型转换后,将相关对象的引用计数交给对方权限管理

我们根据具体的业务逻辑,合理使用上面的三种转换关键字,就可以解决 Core Foundation 对象与 Objective-C 对象相对转换的问题了。

4.其他小知识点:

  • alloc ,new,copy,retain会让引用计数器加1 , 用release,autorelease对引用计数器做减1操作
  • MRC中一定要在delloc中对对象做一次release,然后最后调用super dealloc;ARC中不需要,也不能调用super dealloc
  • ARC
  • 只支持cocoa框架下面的对象,也就是所以继承自NSObjec的类的实例对象
  • 不支持CoreFoundation框架下面的东西,CF的内存管理,需要手动管理,调用CFRelease(<#CFTypeRef cf#>) 和CFRetain(<#CFTypeRef cf#>)等方法管理

5.MRC下写set,init等方法:


- (instancetype)initWithName:(NSString *)name dog:(Dog *)dog
{
    if (self = [super init]) {
        // init方法中不需要判断_name和name是否不同,因为只会在初始化的时候调用一次
        _name = [name copy];
        _dog = [dog retain];
    }
    return self;
}

- (void)setDog:(Dog *)dog
{
    if (_dog != dog) {
        // 因为用不到旧的dog了,所以对旧的dog做一次release,
        [_dog release];
        
        // 要强引用新的dog,对新的dog做一次retain,
        _dog = [dog retain];
    }
}

- (void)setAge:(NSInteger)age
{
    _age = age; // 基本数据类型,不需要自己管理内存
}

- (NSString *)name
{
    return _name;
}

// MRC中一定要在delloc中对对象做一次release,然后最后调用super dealloc;ARC中不需要,也不能调用super dealloc
- (void)dealloc
{
    [_dog release];
    [_name release];
    [super dealloc];

}

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

推荐阅读更多精彩内容

  • 内存管理 简述OC中内存管理机制。与retain配对使用的方法是dealloc还是release,为什么?需要与a...
    丶逐渐阅读 1,948评论 1 16
  • 为什么进行内存管理? 由于移动设备的内存极其有限,所以每个APP所占的内存也是有限制的,当app所占用的内存较多时...
    天天想念阅读 888评论 1 7
  • 29.理解引用计数 Objective-C语言使用引用计数来管理内存,也就是说,每个对象都有个可以递增或递减的计数...
    Code_Ninja阅读 1,470评论 1 3
  • http://www.cnblogs.com/flyFreeZn/p/4264220.html 本文来源于我个人的...
    子键_北京不眠夜阅读 905评论 0 1
  • 最近我利用碎片化时间读了一些书,有投资理财类,有个人管理类的,也有畅销书,共19本,我感觉现在和别人说自己在读书,...
    乐活316阅读 242评论 0 0