内存管理笔记(持续更新...)
这篇纯粹是笔记。
内存管理
堆:需要手动释放。
栈:系统自动管理。
所谓内存管理,就是对内存进行管理,涉及的操作有:
分配内存 :比如创建一个对象,会增加内存占用
清除内存 :比如销毁一个对象,能减小内存占用
内存管理的管理范围
任何继承了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中的内容