MRC生命周期控制机制
唠叨:为啥要复习MRC,ARC已经很智能了,这个思想支使我一直没有认真复习;陌陌面试经历不得让我认真思考这个问题,MRC是ARC的灵魂,不熟悉MRC,ARC就没办法彻底弄明白,有些地方还是会出问题。
仔细体会以下内容:
//引用计数。
NOName *name = [[NOName alloc] init];
//自动调用NoName类中的init方法
// [name retainCount]显示应用计数的值
NSLog(@"RetainCount == %d",[name retainCount]);
[name retain];
NSLog(@"RetainCount == %d",[name retainCount]);
[name release];
NSLog(@"RetainCount == %d",[name retainCount]);
[name release];
//当引用计数显示为0时,自动到NOName类中调用dealloc方法
// Do any additional setup after loading the view, typically from a nib.
//1。当调用alloc ,retain ,copy , new这些方法的时候会引起引用计数加1。
//2。谁创建谁释放(release)。某个实体拥有一个对象时,就要负责对该对象进行清理
//3。alloc ,retain ,copy , new 要跟release配对。
//4。只有alloc ,retain ,copy , new,才需要释放,调用这几个方法的调用者被称为对象的拥有者,拥有者负责释放。
//这个是局部变量,在方法内部要把自己增加的引用去除掉。因为在方法外部根本不可能再访问到这个变量。
UIImageView *imageView = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, 200, 200)];
NSLog(@"imageView retain count %d",[imageView retainCount]);
//会引起引用计数加一(因为害怕狗绳将来一根都没了,所以自己主动加跟狗绳),参照第二条原则,这个引用加一,不需要我们来释放,因为不是我们引起的,uiviewcontroller这个类本身负责释放。
[self.view addSubview:imageView];
NSLog(@"imageView retain count %d",[imageView retainCount]);
//imageview在这个方法内部使用完毕可以释放因为alloc产生的引用计数。
[imageView release];
//全局变量一般要在dealloc中释放。
array = [[NSMutableArray alloc] init];
//不是本身引起引用计数加一,不是拥有者,不负责释放。
// array2 = [NSMutableArray arrayWithCapacity:0]
//这个需要自己释放,因为retain了,你引起了引用计数加一,此处必须retain一下,因为这个对象为自动释放对象(参照 arrayWithCapacity方法的模拟),在将来某个时刻有可能就被释放了,所以自己retain一下,可以保证这个对象不被释放。
array2 = [[NSMutableArray arrayWithCapacity:0] retain];
copy方法利于基于NSCopying方法约定,由各类实现的copyWithZone:方法生成并持有对象的副本,mutableCopy类似。
eg:
注意:copy retain 等方法都有返回值,有些东西没留意到不代表没有
- (instancetype)autorelease OBJC_ARC_UNAVAILABLE;
- (id)copy;
- (id)mutableCopy;
经典题:
Q: MRC手写set方法
A:
-(void)setP:(One *)one{
NSLog(@"setP方法被调用了");
if (p!=one) {//如果one和p的地址是相同的,意味不需要再次赋值了
[p release];//p如果是nil,不执行释放。p如果保存一个对象的地址,先释放掉那个对象
//把原对象p的引用计数增加后,把对象地址给p
p=[one retain];
}
}
Q:autoReleasePool降低内存峰值,
eg、
for (int i = 0; i < 10000; i ++) {
@autoreleasepool{
//1
NoName *obj = [[NoName alloc] init];
//2
// NoName *obj = [NoName changeNoName];
}
}
@implementation NoName
//3
+(id)newNoName
{
NoName *copy1 = [[NoName alloc] init];
return [copy1 autorelease];
}
//4
+(id)changeNoName
{
NoName *copy1 = [[NoName alloc] init];
return [copy1 autorelease];
}
- (void)dealloc{
[super dealloc];
NSLog(@"NoName dealloc");
}
@end
上述代码中
1执行 使用pool没有任何作用,因为临时对象内不包含autoRelease对象
2执行 pool降低内存峰值,原因见changeNoName实现,里面有autorelease对象
小插曲:3 处代码有问题,在new alloc等对象创建类方法中,返回对应的引用计数应为1
搜狐墨客项目中运用autoReleasepool的地方很多,例如如下屏幕截图的地方
先记录这些,查漏补缺吧