• OC内存管理(二)


    一:autorelease

    1> autorelease原理:将我们创建的对象放到一个对象释放池中(是一个栈区)当池子释放时,会将池子中的对象都做一次release操作(自动释放池存放在一个池子中,就近原则,符合先进后出)

    2>自动释放池的创建方式

    (1)ios 5.0以前的创建方式

    NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];

    `````````````````

    [pool  release];//[pool drain];用于mac 

    (2)Ios5.0以后

    @autoreleasepool

    {//开始代表创建自动释放池

    ·······

    }//结束代表销毁自动释放池

    3,类调用类方法中,没有看到引用计数器减一的问题

     例如 NSString *s = [NSString stringWithFormat:@“%d”, 10];这个代码中不需要我们手动释放内存,因为在stringWithFormat:方法内部已经完成了引用计数器减1的操作

    代码:

    好处:不用我们自己管理对象释放的时间
    坏处:  不能灵活的释放我们需要释放对象,如果对象的占用的内存大的话,不要使用autorelease。
    代码示例:
    Person类 
    #import "Person.h"
    #import <Foundation/Foundation.h>
    /**
     *  @class
     注意:继承关系中子类头文件不能用@class
     因为子类要继承父类中的所有属性
     */
    @interface Person : NSObject
    
    @property (nonatomic, retain) NSString *name;
    @property (nonatomic, assign) int age;
    
    - (instancetype)initWithName:(NSString *)name andAge:(int)age;
    + (instancetype)personWithName:(NSString *)name andAge:(int)age;
    
    @end
    @implementation Person
    
    - (instancetype)initWithName:(NSString *)name andAge:(int)age
    {
        if (self == [super init]) {
            
            self.name = name;
            self.age = age;
        }
        return self;
    }
    
    // 注意:也就是在类调用类方法中,alloc中引用计数器加1,因此需要手动释放,当时我们不能在下面调用release
    //      因为刚创建的对象我们就释放,毫无意义,因此我们将它加入到释放池中。由池子来释放
    + (instancetype)personWithName:(NSString *)name andAge:(int)age
    {
        /* 用self调用的原因:
            我们为什么不用类名来调用alloc,因为person子类会继承这个方法,如果用Person的话,子类调用会创建父类的对象
         */
        return [[[self alloc] initWithName:name andAge:age] autorelease];
    }
    
    - (void)dealloc
    {
        self.name = nil;
        NSLog(@"Person ------dealloc");
        [super dealloc];
    }
    
    @end

    Student类继承Person

    #import "Student.h"
    #import "Person.h"
    
    @interface Student : Person
    
    
    @end
    @implementation Student
    
    -(void)dealloc
    {
        NSLog(@"Student------dealloc");
        [super dealloc];
    }
    
    @end

    main函数中

    #import <Foundation/Foundation.h>
    #import "Person.h"
    #import "Student.h"
    int main(int argc, const char * argv[])
    {
        // 按道理说我们在ARC中创建对象的代码必须写在释放池中,因为编译器会将代码加到释放池中
        @autoreleasepool {
        
            Person *p = [Person personWithName:@"zhangsan" andAge:10];
            Student *s = [Student personWithName:@"lisi" andAge:20];
        }
        
        return 0;
    }
              
    二:ARC内存管理机制 (是编译器的特性,手动写的内存管理代码,编译器帮我们完成,不需要我们再去关心)
      1> ARC判断准则:只要没有强指针指向对象,对象就会被释放
      2> 指针的分类
           默认情况下,所有的指针都是强指针,用关键字_strong修饰
           当用__week关键字修饰的指针为弱指针。
          _ _weak Person *p=[[Person alloc]  init];//不合理,对象一创建出来就       被释放掉,对象释放掉后,ARC把指针自动清零。
      3> 特点:默认ARC中不允许调用retain、release、autorelease、retain count,但可以重写dealloc方法,不过该方法中不能写【super  dealloc],因为
    编译器编译时已经帮我们完成了。
           @property的参数:

         Strong:相当于原来的retain(适用于OC对象类型),成员变量是强指针

        Weak:相当于原来的assign,(适用于oc对象类型),成员变量是弱指针

        Assign:适用于非OC对象类型(基础类型)

    ARC也存在循环引用的问题:造成循环引用的原因,dealloc中无法调用,在dealloc中强指针引用的对象无法释放。

    解决的办法:1>头文件中文件包含@class

                       2,property中的修饰符一端strong,一端week,dealloc中不要我们关心,编译器已经帮我们弄好。

    4,将MRC代码转换成ARC代码怎么转换

      步骤:1点击Xcode的 Edit 

               2,选择Edit中的Refactor

               3,选择refactor中的convert to Object-c ARC就可以

      将ARC转回MRC

       步骤:1点击File文件中的Restore snaipshot就可以

    5,OC中的集合的内存管理细节

    • 集合的内存管理细节

    • 1> 当把⼀一个对象添加到集合中时,这个对象会做 了⼀一次retain操作,计数器会+1

    • 2> 当⼀一个集合被销毁时,会对集合⾥里⾯面的所有对 象做⼀一次release操作,计数器会-1

    • 3> 当⼀一个对象从集合中移除时,这个对象会⼀一 次release操作,计数器会-1 

  • 相关阅读:
    Matplotlib绘制漫威英雄战力图,带你飞起来!
    jupyter渲染网页的3种方式
    MySQL全文索引、联合索引、like查询、json查询速度大比拼
    进一步聊聊weight initialization
    深度学习基础(2)
    深度学习基础(1)
    SLAM的前世今生
    深度学习:识别图片中的电话号码(1)
    tf更新tensor/自定义层
    tf训练OTSU
  • 原文地址:https://www.cnblogs.com/-boy/p/4088221.html
Copyright © 2020-2023  润新知