序
这篇文章算是正正经经潜心下来开始重筑基础、探究底层的起点了,之前一直在找这样或那样的借口来逃避学习,消遣生活,如今也着实被各种面试消遣了一个月,感慨良多。总而言之还是希望能通过一些文字,帮助自己和看这篇文章的朋友,理解知识,从而记住知识,到最后形成知识体系,“牵一发,思全身”。也借此帮助自己整理思路,搭建属于自己的知识体系。
第1章 自动引用计数
1.1 什么是自动引用计数
- 自动引用计数(ARC - Automatic Reference Counting)是指在内存管理中 对引用 采用自动计数的技术。
- 时间轴:2012年 - iOS 5 - Xcode 4.2
https://developer.apple.com/download/more/
1.2 内存管理/引用计数
1.2.1 概述
- 办公室开关灯的例子可以很好的描述引用计数对于内存的管理。
- 白天上班人多肯定多开灯,晚上一两个人加班也就开一两盏灯,下班了关灯锁门。
1.2.2 内存管理的思考方式
- 谁生成谁持有,谁引用谁释放,没引用就别想插手别人的人生了(非自己持有的对象无法释放)。
- 对象操作与Objective - C方法的对应
- Cocoa框架中Foundation框架类库的NSObject类担负内存管理的职责。alloc/retain/release/dealloc方法也都指代的是NSObject中对应的实例方法。
1.2.2.1 生成/持有/销毁
这一节尝试代码操作,首先要介绍两个点,一个是ARC和MRC的切换,还有一个是ARC机制下如何观察对象的引用计数(retainCount)。
ARC和MRC切换方法一:单一文件切换
选择项目中的Target-> Build Phases-> Complie Sources中选择需要ARC/MRC的文件双击, ARC在输入框中输入:-fobjc-arc,如果不要ARC则输入:-fno-objc-arc
ARC和MRC切换方法二:切换MRC环境
选择项目中的Target -> Build Sttings -> All -> 搜索‘automatic’ -> 把 Objective-C Automatic Reference Counting 设置为 NO
- ARC机制下如何观察对象的引用计数(retainCount)
printf("retain count = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(obj)));'
这里实际上就是将OC对象转换为CF对象并查看其引用计数,注意这里用的是__bridge,而不是__bridge_retain。
关于对象与指针
- 正文开始前还需要说点基础性的东西,这关系到后面生成/持有的理解问题。
- 我们来看这行代码:
UIView * view = [[UIView alloc] init];
代码很简单,实例化了一个UIView对象,那么谁是对象?如果你认为imgView是对象那就错了,view是指针,指针指向的是对象。- 这行代码实际上产生的是两个东西:UIView变量 和 UIView对象,UIView对象存储于内存,UIView变量指向内存中的对象。
- 本质上来说,类是一种指针类型的变量,程序中定义的UIImageView* 类型只是存放一个地址值,保存在main()函数的动态存储区,它指向实际的UIView对象,而真正的UIView对象则存放堆(heap)内存中。
- 所有对象都只能通过指针变量来访问,堆内存中也可以有多个指针,也就是允许多个指针指向同一个对象,例如:
UIImageView * imgView_other = [[UIImageView alloc] init];
- 如果堆内存中的对象没有任何变量指向该对象,那么程序就无法访问该对象,就要释放内存,否则就造成内存泄漏。
自己生成的对象,自己所持有
- 使用alloc/new/copy/mutableCopy名称开头的方法名意味着自己生成的对象只有自己持有。
- 我们来敲代码试试看:
id obj = [[NSObject alloc] init];
printf("retainCount = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(obj)));
输出结果:retainCount = 1
- 使用NSObject类的alloc类方法就能自己生成并持有对象。指向生成并持有对象的指针被赋值给obj。实际上这就是我上面为什么要谈到的对象与指针的概念,obj是个指针。别着急,下面还有问题。
- copy方法和mutableCopy方法这里就不多赘述了,都是生成并持有对象的副本,区别在于对象的可变和不可变。
非自己生成的对象,自己也能持有
- 用alloc/new/copy/mutableCopy以外的方法获得对象,因为不是自己生成并持有,所以自己不是对象的持有者。
- 同样,敲代码看看:
id obj = [NSMutableArray array];
printf("retainCount = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(obj)));
输出结果:retainCount = 1
- 源代码中,NSMutableArray类对象被赋给obj,但变量obj自己并不持有该对象。使用retain方法可以持有对象。好的,那么问题来了,为什么obj并没有持有对象,打印obj的retainCount,结果依然是1?
这就是上文提及对象与指针的意义,obj其本身是个指针,指向内存中的对象,这里打印的retainCount其实质是对象的retainCount,赋值给obj的是未持有对象的指针。如果想要obj持有对象,就需要:
[obj retain];
printf("retainCount = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(obj)));
输出结果:retainCount = 2
所以这里一定要搞明白,其实这里还可以再探究下array这个类方法,回头有时间再填这个坑吧。
在不需要的时候,将自己持有的对象释放
- 这块原题是“不再需要自己持有的对象时释放”,读起来理解起来实在拗口,改一下。自己持有的对象,一旦不再需要,持有者有义务释放该对象,使用release方法。
- 同样,敲代码:
id obj = [NSMutableArray array];
[obj retain];
printf("retainCount = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(obj)));
[obj release];
printf("retainCount = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(obj)));
输出结果:
retainCount = 2
retainCount = 1