美文网首页
Objective-C高级编程:iOS与OSX多线程和内存管理

Objective-C高级编程:iOS与OSX多线程和内存管理

作者: 丶217 | 来源:发表于2018-01-03 20:42 被阅读0次

    这篇文章算是正正经经潜心下来开始重筑基础、探究底层的起点了,之前一直在找这样或那样的借口来逃避学习,消遣生活,如今也着实被各种面试消遣了一个月,感慨良多。总而言之还是希望能通过一些文字,帮助自己和看这篇文章的朋友,理解知识,从而记住知识,到最后形成知识体系,“牵一发,思全身”。也借此帮助自己整理思路,搭建属于自己的知识体系。

    第1章 自动引用计数

    1.1 什么是自动引用计数
    • 自动引用计数(ARC - Automatic Reference Counting)是指在内存管理中 对引用 采用自动计数的技术。
    • 时间轴:2012年 - iOS 5 - Xcode 4.2
      https://developer.apple.com/download/more/
      image.png
    1.2 内存管理/引用计数
    1.2.1 概述
    • 办公室开关灯的例子可以很好的描述引用计数对于内存的管理。
    • 白天上班人多肯定多开灯,晚上一两个人加班也就开一两盏灯,下班了关灯锁门。
    1.2.2 内存管理的思考方式
    • 谁生成谁持有,谁引用谁释放,没引用就别想插手别人的人生了(非自己持有的对象无法释放)。
    • 对象操作与Objective - C方法的对应
      image.png
    • 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


      image.png
    • ARC和MRC切换方法二:切换MRC环境
      选择项目中的Target -> Build Sttings -> All -> 搜索‘automatic’ -> 把 Objective-C Automatic Reference Counting 设置为 NO


      image.png
    • 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

    相关文章

      网友评论

          本文标题:Objective-C高级编程:iOS与OSX多线程和内存管理

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