内存管理

MRC生命周期控制机制

唠叨:为啥要复习MRC,ARC已经很智能了,这个思想支使我一直没有认真复习;陌陌面试经历不得让我认真思考这个问题,MRC是ARC的灵魂,不熟悉MRC,ARC就没办法彻底弄明白,有些地方还是会出问题。

内存管理_第1张图片
生命周期.png

仔细体会以下内容:

    //引用计数。
   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:


内存管理_第2张图片
copy.png

注意: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的地方很多,例如如下屏幕截图的地方


内存管理_第3张图片
image.png

先记录这些,查漏补缺吧

你可能感兴趣的:(内存管理)