美文网首页
内存管理

内存管理

作者: 愤怒小鸟飞呀飞 | 来源:发表于2018-08-01 14:21 被阅读0次

MRC生命周期控制机制

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

生命周期.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:


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


image.png

先记录这些,查漏补缺吧

相关文章

  • iOS内存管理详解

    目录 block内存管理 autorelease内存管理 weak对象内存管理 NSString内存管理 new、...

  • 第10章 内存管理和文件操作

    1 内存管理 1.1 内存管理基础 标准内存管理函数堆管理函数虚拟内存管理函数内存映射文件函数 GlobalMem...

  • 操作系统之内存管理

    内存管理 包括内存管理和虚拟内存管理 内存管理包括内存管理概念、交换与覆盖、连续分配管理方式和非连续分配管理方式(...

  • JavaScript —— 内存管理及垃圾回收

    目录 JavaScript内存管理内存为什么需要管理?内存管理概念JavaScript中的内存管理JavaScri...

  • OC - OC的内存管理机制

    导读 一、为什么要进行内存管理 二、内存管理机制 三、内存管理原则 四、MRC手动内存管理 五、ARC自动内存管理...

  • 3. 内存管理

    内存管理 内存管理包含: 物理内存管理; 虚拟内存管理; 两者的映射 除了内存管理模块, 其他都使用虚拟地址(包括...

  • Go语言——内存管理

    Go语言——内存管理 参考: 图解 TCMalloc Golang 内存管理 Go 内存管理 问题 内存碎片:避免...

  • jvm 基础第一节: jvm数据区

    程序内存管理分为手动内存管理和自动内存管理, 而java属于自动内存管理,因此jvm的职能之一就是程序内存管理 j...

  • 内存管理

    内存管理的重要性。 不进行内存管理和错误的内存管理会造成以下问题。 内存泄露 悬挂指针 OC内存模型 内存管理是通...

  • 11-AutoreleasePool实现原理上

    我们都知道iOS的内存管理分为手动内存管理(MRC)和自动内存管理(ARC),但是不管是手动内存管理还是自动内存管...

网友评论

      本文标题:内存管理

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