美文网首页
内存管理

内存管理

作者: 金歌漫舞 | 来源:发表于2016-08-06 08:59 被阅读14次

    一.内存基本介绍

    1、OC内存管理的基本概念

         由于移动设备的内存极其有限,所以每个APP所占的内存也是有限制的,当app所占用的内存较多时,系统就会发出内存警告,个app可用的内存是被限制的,如果一个app使用的内存超过一定数量,则系统会向该app发送Memory Warning消息。收到此消息后,需要回收一些不需要再继续使用的内存空间,比如回收一些不再使用的对象和变量等,否则程序会崩溃​
    

    2、OC内存管理的范围​
    管理范围:管理任何继承自NSObject的对象​

    每个OC对象都有自己的引用计数器,是一个整数表示对象被引用的次数,即现在有多少东西在使用这个对象。对象刚被创建时,默认计数器值为1,当计数器的值变为0时,则对象销毁。

    1).在每个OC对象内部,都专门有8个字节的空间来存储引用计数器  那么为什么是8个字节?  OC为提供了一个获取引用计数器的方法  -(NSUInteger)retainCount OBJC_ARC_UNAVAILABLE  这个方法的返回值为:NSUInteger  接着来看NSInteger的定义  typedef NSUInteger unsigned long  到这儿就清楚了,因为数据类型unsigned long的长度就是8个字节

    )、引用计数器的作用

    2). 引用计数器是判断对象要不要回收的依据就是计数器是否为0,若为0则回收,不为0则不回收

    3).对引用计数器的操作
      给对象发送消息,进行相应的计数器操作。

    retain消息:使计数器+1,该方法返回对象本身  语法:[对象名 retain]

    release消息:使计数器-1(并不代表释放对象)  语法:[对象名 release]

    retainCount消息:获得对象当前的引用计数器值 %lu

    4)、对象的销毁​

    ​ 总结:我们可以在跟据dealloc方法是否被调用得知一个对象是不是被释放了。

    注意:永远不要直接通过对象调用dealloc方法(实际上调用并不会出错)
      一旦对象被回收了, 它占用的内存就不再可用, 坚持使用会导致程序崩溃(野指针错误)为 了防止调用出错,可以将“野指针”指向nil(0)

    重写dealloc方法的代码规范​

     1. 一定要[super dealloc],而且要放到最后,意义是:你所创建的每个类都是从父类, 根类继承来的,有很多实例变量也会继承过来,这部分变量有时候会在你的程序内使用,它们不会自动释放内存,你需要调用父类的dealloc方法来释放,然而在此之前你需要先把自 己所写类中的变量内存先释放掉,否则就会造成你本类中的内存积压,造成泄漏  2. 对self(当前)所拥有的的其他对象做一次release操作
    
    -(void)dealloc
    {
        [_car release];
        [super dealloc];
    
    
    }
    
    

    5)对象所有权
    对象所有权的概念
      任何对象都可能拥有一个或多个所有者,只要一个对象至少还拥有一个所有者,它就会继续存在

    6)、注意事项
    如果对象的计数器不为0,那么在整个程序运行过程,它占用的内存就不可能被回收(除非整个程序已经退出 )

    任何一个对象,刚生下来的时候,引用计数器都为1。(对象一旦创建好,默认引用计数器就是1)当使用alloc、new或者copy创建一个对象时,对象的引用计数器默认就是1

    3、OC内存管理分类

    OC中的3种内存管理方式
    Mannul Reference Counting(MRC,手动管理,在开发iOS5.0之前的版本的项目时我们要自己负责使用引用计数来管理内存,比如要手动retain、release、autorelease 等,而在其后的版本可以使用ARC,让系统自己管理内存。)

    Automatic Reference Counting(ARC,自动引用计数,iOS5.0 之后推出的)

    Garbage Collection(垃圾回收)。iOS不支持垃圾回收; ARC作为苹果新 供的技术,苹果推荐开发者使用ARC技术来管理内存;

    二、手动内存管理快速入门​

    1、关闭ARC的方法(其中之一)

    三、内存管理的原则
    1、内存管理的原则

    原则
    如果你通过alloc,new,copy来创建了一个对象,那么你就必须对应的调用release或者 autorelease方法
    当产生一个新的引用的时候, 需要将对象引用计数器 +1, 即调用对象的 retain 方法Person *p2 = [p1 retain]
    谁让对象retain的,就由谁release,即当某个引用不再指向该对象的时候,就需要将该对象的引用计数器-1,
    总结
      有始有终,有加就应该有减。谁创建,谁release,谁retain,谁release。  曾经让某个对象计数器加1,就应该让其在最后-1(让引用计数器复原到使用之前的状态).

    little tips:  一般都写完一个加操作,我们就会对应的写下减操作,这样会保证我们不会写完程序后有遗漏,造成对象不能释放。

    2、内存管理研究内容(画图理解)

    1)野指针(僵尸对象)
      僵尸对象: 已经被销毁的对象(不能再使用的对象)  野指针:指向僵尸对象(不可用内存)的指针

    2)内存泄露
    四、单个对象内存管理
    1、避免使用僵尸对象的方法​

    为了防止不小心调用了僵尸对象,可以将指针赋值nil(对象的空值)

    空指针:没有指向任何东西的指针,给空指针发送消息不会报错​

    关于nil和Nil及NULL的区别:

    nil: A null pointer to an Objective-C object. nil 是一个OC对象值

    Nil: A null pointer to an Objective-C class.给类对象赋值​

    NULL: A null pointer to anything else, is for C-style memory pointers. 用于对非对象指针赋空值​

    2、对象的内存泄露

    1)加1操作 和 减1操作 个数不匹配,导致内存泄露
    2)对象使用的过程中被赋值了nil,导致内存泄露​

    五​、多个对象内存管理
    1)基本数据类型:直接赋值​

    2)OC对象类型​
    六、​@property参数(一)
    1、@property

    @property Xcode4.4前

    1、@property + 手动实现  2、@property int age; + @synthesize age; //get和set方法的声明和实现都帮我们做了

    @property Xcode4.4增强   @property int age;  1、生成_age  2、生成_age的get和set方法的声明  3、生成_age的get和set方法的实现

    2、@property 参数
    格式:@property (参数1,参数2) 数据类型 方法名​

    1. retain​:对象release旧值,retain新值(适用于OC对象类型

    2. ​assign:直接赋值(默认,适用于非oc对象类型)

    3. copy​:release旧值,copy新值

    3、@property 参数(二)

    1)是否要生成set方法(若为只读属性,则不生成)
    readonly:只读,只会生成getter的声明和实现readwrite:默认的,同时生成setter和getter的声明和实现

    2)多线程管理(苹果在一定程度上屏蔽了多线程操作)
    nonatomic:高性能,一般使用这个atomic:低性能,默认

    atomic是Objc使用的一种线程保护技术,基本上来讲,是防止在写未完成的时候被另外一个线程读取,造成数据错误。而这种机制是耗费系统资源的,所以在iPhone这种小型设备上,如果没有使用多线程间的通讯编程,那么nonatomic是一个非常好的选择。

    3)set和get方法的名称

    修改set和get方法的名称,主要用于布尔类型。因为返回布尔类型的方法名一般以is开头,修改名称一般用在布尔类型中的getter。

    @property(nonatomic,assign, setter=abc:,getter=haha)int age
    可以理解为把[p setAge: ]------> [p abc:], [p age] ---------> [p haha];p.age 不会报错(内部优化)

    @property(nonatomic,assign, setter=setVip:,getter=isVip) BOOL vip;

    七、@class的使用​
    作用

    1、可以简单地引用一个类简单使用@class Dog; //类的引入仅仅是告诉编译器: Dog是一个类; 并不会包含Dog这个类的所有内容

    具体使用
    在.h文件中使用@class引用一个类在.m文件中使用#import包含这个类的.h文件

    为了简单起见:A类是引用类,B类是被引用类,这里先不考虑A类的实现文件。通常引用一个类有两种办法:一种是通过#import方式引入;另一种是通过@class引入;​

    2、@class和#import的区别

    这两种的方式的区别在于:1)#import方式会包含被引用类的所有信息,包括被引用类的变量和方法;@class方式只是告诉编译器在A.h文件中 B *b 只是类的声明,具体这个类里有什么信息,这里不需要知道,等实现文件中真正要用到时,才会真正去查看B类中信息;

    2)使用@class方式由于只需要被引用类(B类)的名称就可以了,而在实现类由于要用到被引用类中的实体变量和方法,所以需要使用#import来包含被引用类的头文件;

    3)通过上面2点也很容易知道在编译效率上,如果有上百个头文件都#import了同一个文件,或者这些文件依次被#improt(A->B, B->C,C->D...),一旦最开始的头文件稍有改动,后面引用到这个文件的所有类都需要重新编译一遍,这样的效率也是可想而知的,而相对来讲,使用@class方式就不会出现这种问题了;

    所以:我们实际开发中尽量在.h头文件中使用@class

    作用上的区别

    import会包含引用类的所有信息(内容), 包括引用类的变量和方法 @class仅仅是告诉编译器有这么一个类, 具体这个类里有什么信息, 完全不知道

    效率上的区别
    如果有上百个头文件都#import了同一个文件,或者这些文件依次被#import,那么一旦最开始的头 文件稍有改动,后面引用到这个文件的所有类都需要重新编译一遍 , 编译效率非常低 相对来讲,使用@class方式就不会出现这种问题了

    八、循环retain问题​

    ​循环retain的场景

    比如A对象retain了B对象,B对象retain了A对象循环retain的弊端这样会导致A对象和B对象永远无法释放

    循环retain的解决方案
    当两端互相引用时,应该一端用retain、一端用assign

    九、autorelease基本使用​

    1、自动释放池及autorelease介绍
    自动释放池
    (1)在iOS程序运行过程中,会创建无数个池子,这些池子都是以栈结构(先进后出)存在的。(2)当一个对象调用autorelease方法时,会将这个对象放到位于栈顶的释放池中

    自动释放池的创建方式

    (1)iOS 5.0以前的创建方式

    NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];
    ............
    [pool release];//[pool drain];用于mac
    (2)iOS5.0以后

    @autoreleasepool{//开始代表创建自动释放池·······
    }//结束代表销毁自动
    autorelease
    是一种支持引用计数的内存管理方式它可以暂时的保存某个对象(object),然后在内存池自己的排干(drain)的时候对其中的每个对象发送release消息注意,这里只是发送release消息,如果当时的引用计数(reference-counted)依然不为0,则该 对象依然不会被释放。可以用该方法来保存某个对象,也要注意保存之后要释放该对象。

    2、为什么会有autorelease?

    OC的内存管理机制中比较重要的一条规律是:谁申请,谁释放考虑这种情况,如果一个方法需要返回一个新建的对象,该对象何时释放?

    方法内部是不会写release来释放对象的,因为这样做会将对象立即释放而返回一个空对象;调用者也不会主动释放该对象的,因为调用者遵循“谁申请,谁释放”的原则。那么这个时候,就发生了内存泄露。

    针对这种情况,Objective-C的设计了autorelease,既能确保对象能正确释放,又能返回有效的对象。

    使用autorelease的好处
    (1)不需要再关心对象释放的时间(2)不需要再关心什么时候调用release

    3、autorelease基本用法
    基本用法
    (1)会将对象放到一个自动释放池中(2)当自动释放池被销毁时,会对池子里的所有对象做一次release(3)会返回对象本身(4)调用完autorelease方法后,对象的计数器不受影响(销毁时影响)

    在autorelease的模式下,下述方法是合理的,即可以正确返回结果,也不会造成内存泄露

    3、autorelease是什么原理?

    autorelease实际上只是把对release的调用延迟了,对于每一个Autorelease,系统只是把该 Object放入了当 前的Autorelease pool中,当该pool被释放时,该pool中的所有Object会被调用 Release。

    4、autorelease何时释放?
    对于autorelease pool本身,会在如下两个条件发生时候被释放1)手动释放Autorelease pool2)Runloop结束后自动释放

    对于autorelease pool内部的对象在引用计数的retainCount == 0的时候释放。

    release和autorelease pool 的 drain都会触发retain--事件。

    相关文章

      网友评论

          本文标题: 内存管理

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