一、 内存管理
1 管理方式:MRC(已经很少用) ARC(常用)
ARC:称为自动引用计数,ARC是iOS5推出的功能,通过ARC系统可以自动管理内存。在ARC下只要所有的强引用(指针)引用对象,那么该对象就会被释放。
注意: ARC下不允许使用retain、release、retainCount等MRC下使用的方法,如果使用dealloc方法时不允许调用[super dealloc]
二、与内存相关的修饰符
- strong:强引用,使用后该对象的‘计数器’+1。
- weak: 弱引用,使用后该对象的‘计数器’引用计数为1,当没有其他强指针引用该修饰符修饰的对象后,该对象被置为nil,weak可以有效的避免野指针。
- readwrite: 可读可写修饰符,需要生成setter和getter时使用,一般省略。
- readonly: 只读,只会生成getter不会生成setter,如果对该修饰符修饰的对象赋值会报错。
- assign: 赋值特性,不涉及引用计数,弱引用。setter方法讲传入参数赋值给实例变量。
- retain:表示持有特性,setter方法将传入参数先保留,再赋值,‘计数器’+1
- copy:表示拷贝特性,setter方法传入的对象会被复制一份,需要生成新的对象时使用。一般用于字符串、NSMultebleArray等。
- nonatomic: 非原子操作,不加同步,非线程安全。对线程访问时可以提高性能。决定编译器生成setter和getter是否是原子操作。
- atomic: 原子操作,同步的,线程安全,与nonatomic相反
- weak vs assign
weak:在ARC中,在有可能出现循环引用的时候,往往通过让其中一端将强引用变为弱引用来解决问题,防止多次强引用时也需要用到weak。
weak是表示该属性定义了一种‘非拥有关系’。这种属性设置新值时,设置方法既不保留新值也不释放旧值。当属性值所指向的对象遭到销毁时,属性值也会清空。assing的‘设置方法’只会用于‘纯量类型’的简单赋值操作(例如CGFloat、NSInteger等)
assign可以用于非OC对象,而weak必须用于OC对象
三、Autorelease
autorelease是一种支持引用计数的内存管理方式,autorelease实际上只是把对release的调用延迟了。对于每个autorelease,系统只是把该对象放入了当前的autorelease pool中,当该pool释放时,该pool中的所有对象会被调用release
- autorelease何时释放: 1,手动释放 2,runloop结束后自动释放
- 原理:程序在main()调用的时候会自动创建一个autorelease,然后在每一个runloop,系统会隐式创建一个autorelease pool,这样所有的release pool会构成一个栈式结构,每一个runloop结束时,当前栈顶的autorelease pool会被销毁,这样pool里的每个对象都会被release
注意: ARC下之能使用@ autoreleasepool block,尽量避免对大内存使用该方法。不要把大量的循环操作放到统一个autorelease pool中,这样会造成内存峰值。
问题:autorelease pool 什么时候释放
- 在没有手动加autorelease pool的情况下,autorelease对象是在当前的runloop迭代结束时释放的,它能够释放的原因是系统在每个runloop迭代中都加入了autoreleasepool push和pop
四、内存泄露(memory leak)
- oc与c混编,c使用的内存没有被释放
- block或者其他原因导致循环引用会产生内存泄露
- NSTimer对象会对它的target进行retain,从而保证了timer调用时的正确性,但是又会引入了接收者的内存管理问题,特别是对于重复性的timer,他所引用的对象一直存在,这将会导致内存泄露。解决的办法是,NSTimer提供了一个invalidate的方法,在执行完invalidate方法后timer会变无效。
五、循环引用
- block在执行copy时会对block内部用到的对象进行强引用,当某个类将block作为自己的全局变量强引用,然后该类在block里有使用了该类本身,它们之间会形成一个强引用环,都无法释放,就会导致循环引用。
- 解决办法:
__weak typeof(self) weakSelf=self;
在block里使用weakSelf - 在使用代理模式的时候要注意Delegate一定是weak的,否则会导致代理和原对象之间形成强引用循环导致循环引用。
- 解决办法:
六、OC对象与CF对象的桥接
- __bridge只做OC和CF对象之间的类型转换不涉及对象的所有权的转化
- __bridge_transfer:用在将CF对象转换成OC对象时,并且将CF对象的所有权交给OC对象,此时ARC就能自动管理内存了。作用同于CFBridgeTranfer()
- __bridge_retainde:用在将OC对象转换成CF对象,并且将OC对象的所有权交给CF对象来管理,作用同于CFBridgeRetain()。此时我们要使用完CF对象后用CFRelease将其手动释放。
NSURL *url = [NSURL URLWithString:@"http://www.baidu.com"];
CFURLRef urlRef = (__bridge_retained CFURLRef)url;
CFRelease(urlRef);