美文网首页iOS小筑
iOS开发之属性标识符copy的抉择

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

作者: 华子小筑 | 来源:发表于2016-12-08 14:32 被阅读57次

    问题

    当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的抉择

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