iOS开发之属性标识符copy的抉择

问题

当NSString/NSArray/NSDictionary类型时,属性标识符你使用copy or strong?

对象接收到copy消息会触发怎样的动作

对象接收到copy消息,如果对象的类型已经遵守了NSCopying协议那么会触发- (id)copyWithZone:(nullable NSZone *)zone;如果未遵守该协议那么-[User copyWithZone:]: unrecognized selector sent to instance 0x100203ed0

NSCopying协议的内容

@protocol NSCopying
//Returns a new instance that’s a copy of the receiver.
- (id)copyWithZone:(nullable NSZone *)zone;
@end

收到copy消息的对象都会创建一个新的实例对象吗?

- (void)testMethod {
        #pragma mark - NSString
        NSString* k_name = [NSString stringWithFormat:@"%@",@"HZiOS"];
        NSLog(@"(1)%@---%p",k_name,k_name);
        NSString* another_name = [k_name copy];
        NSLog(@"(2)%@---%p",another_name,another_name);
        
        #pragma mark - NSMutableString
        NSMutableString* kmutable_name = [NSMutableString stringWithFormat:@"%@",@"HZiOSmutable"];
        NSLog(@"(3)%@---%p",kmutable_name,kmutable_name);
        NSString* other_name = [kmutable_name copy];
        NSLog(@"(4)%@---%p",other_name,other_name);
        /*
         (1)HZiOS---0x534f695a4855
         (2)HZiOS---0x534f695a4855
         (3)HZiOSmutable---0x100207120
         (4)HZiOSmutable---0x100700110
         */
}
NSString不可变字符串

上述代码中NSString类型的变量在收到copy消息的时候并未返回一个全新的实例对象,可以看到两个变量的内存地址是一样的;由此可以猜想在NSString类型的变量收到copy消息时在- (id)copyWithZone:(nullable NSZone *)zone;方法中做了逻辑判断,[k_name isKindOfClass:[NSString class]]此条件成立则直接返回当前实例对象;

NSMutableString可变字符串

上述代码中NSMutableString的变量kmutable_name收到copy消息返回了一个不可变NSString的变量other_name;两个变量的内存地址不一致;这就说明重新创建一个全新的实例变量;

为什么NSString和NSMutableString接收到copy消息操作会不一样呢?
  • NSString类型接收到copy消息如果返回一个新实例,那么内存中就存在两块内存空间存储相同值的不可变字符串,这样做的话会浪费本来就很稀缺的内存空间,所以具体实现上apple的工程师应该做一些优化;
  • NSMutableString类型接收到消息返回一个新实例,原因在于它返回的是一个不可变对象,必须是一个新的实例,如何返回内存地址一致那么后期的操作上会引发异常;

NSArray类型成员变量的修饰符属性你使用copy or strong?

答案是使用copy;为什么使用copy这个就是为了完全避免被子类类型的变量赋值后导致的程序异常;可以试验将下列代码中的属性标识符copy换成strong;看一下运行的结果;

#import "HZPersonCopy.h"
@interface HZPersonCopy ()
@property(nonatomic,copy)NSArray* array;
@end
@implementation HZPersonCopy

-(void)test{
     self.array = @[@"1",@"2"];
     NSMutableArray* mutableArray =[@[@"1",@"2",@"3"] mutableCopy];
    self.array = mutableArray;
    NSLog(@"--->:%@",self.array[2]);
    [mutableArray removeLastObject];
    NSLog(@"--->:%@",self.array[2]);
 }

总结:

当属性类型为NSString,并且赋值的类型也是NSString(对于不可变类型的集合同样适用)那么使用copy和strong没有区别,但还是建议使用copy;
-(void)testCopyStr{
        #pragma mark - copy属性
        NSString* k_name = [NSString stringWithFormat:@"%@",@"HZiOS"];
        NSLog(@"%@---%p",k_name,k_name);
        self.name = k_name;
        NSLog(@"%@---%p",self.name,self.name);
        k_name = [NSString stringWithFormat:@"%@",@"anotherStr"];
        NSLog(@"%@---%p",k_name,k_name);
        NSLog(@"%@---%p",self.name,self.name);
        
        #pragma mark - test copy and mutableCopy
        self.name = [k_name copy];
        NSLog(@"%@---%p",self.name,self.name);
        self.name = [k_name mutableCopy];
        NSLog(@"%@---%p",self.name,self.name);
        NSLog(@"----------------------------------");
    
        /*
          HZiOS---0x534f695a4855
          HZiOS---0x534f695a4855
          anotherStr---0x10b192a02d085a5
          HZiOS---0x534f695a4855
          anotherStr---0x10b192a02d085a5
          anotherStr---0x10b192a02d085a5
         */
}

-(void)testStrongStr{
    #pragma mark - strong属性
    NSString* k_name = [NSString stringWithFormat:@"%@",@"HZiOS1"];
    NSLog(@"%@---%p",k_name,k_name);
    self.anotherName = k_name;
    NSLog(@"%@---%p",self.anotherName,self.anotherName);
    k_name = [NSString stringWithFormat:@"%@",@"anotherStr1"];
    NSLog(@"%@---%p",k_name,k_name);
    NSLog(@"%@---%p",self.anotherName,self.anotherName);
    
   #pragma mark - test copy and mutableCopy
    self.anotherName = [k_name copy];
    NSLog(@"%@---%p",self.anotherName,self.anotherName);
    self.anotherName = [k_name mutableCopy];
    NSLog(@"%@---%p",self.anotherName,self.anotherName);
    
    /*
      HZiOS1---0x1002036a0
      HZiOS1---0x1002036a0
      HZiOS1addStr---0x1002036a0
      HZiOS1addStr---0x1002036a0
      HZiOS1addStr---0x1001029b0
      HZiOS1addStr---0x100700150
     */
}
唯一需要注意的是属性类型为NSString时,接收到[xxx mutableCopy]的返回值,使用copy和strong会略有不同;这个可以看代码中的内存地址;

copy属性特征:接收到[xxx mutableCopy]的返回值,实际又接收了copy消息,这样在内存中有存在两块内存空间存储相同的不可变字符,so优化合并,所以打印出来的内存地址是一样的;

你可能感兴趣的:(iOS开发之属性标识符copy的抉择)