美文网首页
iOS Objective-C中strong和copy

iOS Objective-C中strong和copy

作者: 王_凯 | 来源:发表于2017-06-23 15:12 被阅读0次

    我们在开发中声明NSString,NSArray,NSDictionary ,block等一般都用copy来修饰,但是为什么要这么做,是不是非用copy不可,用strong行不行?很多iOS面试官也会问这个问题,那我就根据我自己的经验来说说这个问题,希望多多指正批评。

    首先来了解一下OC的内存管理。

    说到Strong关键字那就要了解了解ARC(Automatic Reference Counting)之前和ARC之后的内存管理。
    OC内存管理遵循“谁创建,谁释放;谁引用,谁管理”的机制。当创建或引用一个对象的时候,需要向它发送alloc copy retain 消息,当释放该对象时需要发送release消息,当该对象引用计数为0时,系统将释放该对象,这是OC的手动内存管理机制;
    iOS5.0之后OC又提供了自动管理机制ARC,管理机制跟手动管理机制一样,只是不再需要手动调用 retain release autorelease;它是 编译时特性,当你启用ARC时,在适当的位置插入release和autorelease;它引用了strong和weak 关键字,strong修饰的指针变量指向对象时,当指针指向新值,或者指针不再存在时,相关联的对象就会自动释放;而weak修饰的指针变量指向对象,当对象的拥有者指向新值或者不存在时weak修饰的指针则自动置为nil,这是ARC管理机制
    想深入了解内存管理推荐一本《Objective-C高级编程》

    言归正传

    也就是说Strong和ARC之前的retain一样都是使引用计数+1。
    如果用strong修饰,不管属性是可变类型还是不可变类型,指针都会指向通一块内存,例如

    @interface BaseViewController ()
    
    @property(nonatomic,strong)NSArray *array;
    @property(nonatomic,strong)NSMutableArray *marray;
    
    @end
    
    @implementation BaseViewController
    - (void)viewDidLoad {
        [super viewDidLoad];
        NSMutableArray *arraya = [[NSMutableArray alloc] initWithObjects:@"1", nil];
        self.array = arraya;
        self.marray = arraya;
        [arraya addObject:@"2"];
        NSLog(@"1-%p-%@",self.array,self.array);
        NSLog(@"2-%p-%@",self.marray,self.marray);
        NSLog(@"3-%p-%@",arraya,arraya);
    }
    @end
    

    打印结果

    1-0x600000054010-(
        1,
        2
    )
    2-0x600000054010-(
        1,
        2
    )
    3-0x600000054010-(
        1,
        2
    )
    

    先来了解下MutableCopy与Copy
    对系统非容器类不可变对象调用Copy方法其实只是把当前对象的指针指向了原对象的地址,而调用mutableCopy方法则是新分配了一块内存区域并把新对象的指针指向了这块区域。对于可变对象来说调用Copy和MutableCopy方法都会重新分配一块内存。但是copy和mutableCopy的区别在于copy在复制对象的时候其实是返回了一个不可变对象,因此当调用方法改变对象的时候会崩溃
    容器对象和非容器对象在分别调用Copy和MutableCopy时没有什么分别,不可变对象调用Copy方法只是增加了对原对象的指针的引用,调用MutableCopy方法是重新分配一块内存,然后把新对象指向新内存。而对于可变对象不管调用Copy还是MutableCopy都是新分配一块内存。但是虽然重新分配了一块内存,但是对象里面的数据依然是指针复制的

    情景一

    #import "BaseViewController.h"
    #import "TestModel.h"
    
    @interface BaseViewController ()
    
    @property(nonatomic,copy)NSArray *array;
    
    @end
    
    @implementation BaseViewController
    - (void)viewDidLoad {
        [super viewDidLoad];
        NSArray *arraya = [[NSArray alloc] initWithObjects:@"1", nil];
        self.array = arraya;
        NSLog(@"1-%p-%@",self.array,self.array);
        NSLog(@"3-%p-%@",arraya,arraya);
    }
    @end
    

    输出结果

    1-0x60800000c970-(
        1
    )
    3-0x60800000c970-(
        1
    )
    

    由此可见这种情况下其实和strong效果一样

    情景二

    #import "BaseViewController.h"
    #import "TestModel.h"
    
    @interface BaseViewController ()
    
    @property(nonatomic,copy)NSArray *array;
    
    @end
    
    @implementation BaseViewController
    - (void)viewDidLoad {
        [super viewDidLoad];
        NSMutableArray *arraya = [[NSMutableArray alloc] initWithObjects:@"1", nil];
        self.array = arraya;
        NSLog(@"1-%p-%@",self.array,self.array);
        NSLog(@"3-%p-%@",arraya,arraya);
    }
    @end
    

    输出结果

    1-0x600000200750-(
        1
    )
    3-0x60000005f890-(
        1,
        2
    )
    

    由此可见NSMutableArray过copy之后重新分配了一块内存并由self.array指向改内存,而且self.array依然是不可变的

    情景三

    #import "BaseViewController.h"
    #import "TestModel.h"
    
    @interface BaseViewController ()
    
    @property(nonatomic,copy)NSMutableArray *array;
    
    @end
    
    @implementation BaseViewController
    - (void)viewDidLoad {
        [super viewDidLoad];
        NSMutableArray *arraya = [[NSMutableArray alloc] initWithObjects:@"1", nil];
        self.array = arraya;
        [arraya addObject:@"2"];
    //    [self.array addObject:@"3"];//程序崩溃
        NSLog(@"1-%p-%@",self.array,self.array);
        NSLog(@"3-%p-%@",arraya,arraya);
    }
    @end
    

    输出结果

     1-0x60800001fd60-(
        1
    )
    3-0x60800004fe10-(
        1,
        2
    )
    

    由此可见NSMutableArray过copy之变成了不可变数组,重新分配了一块内存并由self.array指向,self.array虽然声明的是可变数组但其实已经变成了不可变数组,所以调用addObject方法会崩溃

    总结:用copy还是用strong要根据自己的需求来决定

    相关文章

      网友评论

          本文标题:iOS Objective-C中strong和copy

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