美文网首页应用iOSOC内存
内存管理笔记(持续更新...)

内存管理笔记(持续更新...)

作者: HHHHHHHHHHD | 来源:发表于2015-10-21 23:42 被阅读1989次

这篇纯粹是笔记。

内存管理

堆:需要手动释放。
栈:系统自动管理。

所谓内存管理,就是对内存进行管理,涉及的操作有:

分配内存 :比如创建一个对象,会增加内存占用
清除内存 :比如销毁一个对象,能减小内存占用

内存管理的管理范围

任何继承了NSObject的对象
对其他非对象类型无效(int、char、float、double、struct、enum等)

只有OC对象才需要进行内存管理的本质原因

OC对象存放于堆里面
OC基础类型一般放在栈里面(栈内存会被系统自动回收)

引用计数器的常见操作

给对象发送一条retain消息,可以使引用计数器值+1(retain方法返回对象本身)
给对象发送一条release消息,可以使引用计数器值-1
给对象发送retainCount消息,可以获得当前的引用计数器值

需要注意的是: release并不代表销毁\回收对象,仅仅是计数器-1。
空指针\野指针

僵尸对象
已经被销毁的对象(不能再使用的对象)

野指针
指向僵尸对象(不可用内存)的指针
给野指针发消息会报EXC_BAD_ACCESS错误

空指针
没有指向存储空间的指针(里面存的是nil,也就是0)
给空指针发消息是没有任何反应的。
为了避免野指针错误的常见办法
在对象被销毁之后,将指向对象的指针变为空指针

delloc方法注意细节

当一个对象的引用计数器值为0时
这个对象即将被销毁,其占用的内存被系统回收
系统会自动给对象发送一条dealloc消息
(因此,从dealloc方法有没有被调用,就可以判断出对象是否被销毁)
dealloc方法的重写
一般会重写dealloc方法,在这里释放相关资源,dealloc就是对象的遗言
一旦重写了dealloc方法,就必须调用[super dealloc],并且放在最后面调用
使用注意
不能直接调用dealloc方法
一旦对象被回收了,它占用的内存就不再可用,坚持使用会导致程序崩溃(野指针错误)

Xcode开启僵尸对象的监控
截图1.png
苹果官方规定的内存管理原则

谁创建谁release:如果你通过alloc、new或[mutable]copy来创建一个对象,那么你必须调用release或autorelease。
谁retain谁release:只要你调用了retain,就必须调用一次release。
总结一下就是
有加就有减
曾经让对象的计数器+1,就必须在最后让对象计数器-1

set方法的内存管理
- (void)setCar:(Car*)car
{
   if(car!= _car)
   {
        // 对当前正在使用的车(旧车)做一次release
        [_car release];
        
        // 对新车做一次retain操作
            _car=[car retain];
   }
}
dealloc方法内存管理
- (void)dealloc
{
     //当人不在了,代表不用车了
     // 对车做一次release操作
   [_car release]; 
     //调用dealloc方法时候一定要调用父类的dealloc的方法
   [super dealloc];
}

@property参数修饰
控制set方法的内存管理

retain:release旧值,retain新值(用于OC对象)
assign:直接赋值,不做任何内存管理(默认,用于非OC对象类型)
copy :release旧值,copy新值(一般用于NSString*,block)

控制需不需生成set方法

readwrite:同时生成set方法和get方法(默认)
readonly :只会生成get方法

多线程管理

atomic :性能低(默认)
nonatomic:性能高

控制set方法和get方法的名称

setter:设置set方法的名称,一定有个冒号:
getter:设置get方法的名称

强引用strong与弱引用weak的广义区别:

强引用strong也就是我们通常所讲的引用,其存亡直接决定了所指对象的存亡。如果不存在指向一个对象的引用,并且此对象不再显示列表中,则此对象会被从内存中释放。
弱引用weak除了不决定对象的存亡外,其他与强引用相同。即使一个对象被持有无数个若引用,只要没有强引用指向他,那麽其还是会被清除。没办法,还是 “强哥” 有面子。

简单讲strong等同retain
ARC时候用strong,MRC时候用return
weak比assign多了一个功能,当对象消失后自动把指针变成nil,好处不言而喻。所以,我觉得在delegate时候用weak会好过assign。

至于block的话,就要结合考虑。(weak,copy,assign)这三个结合情况,在下一篇的介绍。

__weak ,__strong ,__unsafe_unretained,** __autoreleasing**用来修饰变量.
(__strong) 是缺省的关键词。
(__weak) 声明了一个可以自动 nil 化的弱引用。
(__unsafe_unretained) 声明一个弱应用,但是不会自动nil化,也就是说,如果所指向的内存区域被释放了,这个指针就是一个野指针了。
(__autoreleasing) 用来修饰一个函数的参数,这个参数会在函数返回的时候被自动释放。

@class和#import的区别

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

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

MRC循环retian

循环retain的场景
比如A对象retain了B对象,B对象retain了A对象

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

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

同理,在ARC下,如果两个对象相互引用,并且用strong修饰,那两个对象都无法释放。解决方案:其中一个对象用weak修饰,一个用strong修饰。那就可以让两个对象都可以释放。

nil Nil NULL NSNull 之间的区别

nil 是对objective c id 对象赋空值
Nil: 表示对类进行赋空值
NULL: 用于对非对象指针赋空值,比如C指针
NSNull 对于像NSArray这样的类型,nil或NULL不能做为加到其中的Object,如果定义了一个NSArray,为其分配了内存,又想设置其中的内容为空,则可以用[NSNULL null】返回的对象来初始化NSArray中的内容

相关文章

网友评论

  • js_2013:说的真好,之前对内存管理不理解,看了此文章又有点懂了
    清蒸鱼跃龙门:@js_2013 那swift怎么管理内存??
    js_2013:@像孩子一样 swift里面好像没有这些概念了😁
    HHHHHHHHHHD:@js_2013 内存管理博大精深。虽然现在是ARC百编程,但是内存管理一样不可忽视。

本文标题:内存管理笔记(持续更新...)

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