Property copy

2018-02-01  本文已影响10人  struggle3g

概念

    Property修饰关键字copy

     创建一个引用计数为1的一个对象,释放旧对象,然后赋值,此属性只对那些实行了NSCopying协议的对象类型有效。copy方法默认是复制该对象的不可变副本。

    copy的setrer方法

/** 第一种setter方法  ARC  模式下

@param Name 输入值 */

- (void)setName:(NSString *)Name{

    if(_Name != Name){

        _Name = nil;

        _Name = [Name copy];

    }

}

/** 第二种setter方法

@param Name 输入值 */

- (void)setName:(NSString *)Name{

        _Name = [Name copy];

}

    上述两种方法,根据概念实现setter方法的时候选择第一个对象是合适的,判断这个对象的值是否是该属性变量 的值,不是,首先让旧属性的变量置nil,ARC模式下,会直接将这个变量清空,释放,然后在复制一份输入值给这个属性的变量。

    然后根据我写过的深、浅拷贝来进行分析。

    为什么NSString会用copy修饰而不用Strong去修饰,首先先看我的另一片关于Strong的文章,里面介绍了strong修饰属性的问题。

    下面是copy需要注意的,首先先实现一个实例。

@interface CopyClass : NSObject

@property(copy,nonatomic)NSString *Name;

@property(copy,nonatomic)NSMutableString *MutableName;

@end

    将一个可变的NSMutableString test1赋值给Name、MutableName,再将test1的值改变,分别打印出来test1、Name、MutableName的内存地址。分析

NSMutableString *test1 = [[NSMutableString alloc]initWithString:@"test"];

CopyClass *testClass = [[CopyClass alloc]init];

testClass.Name = test1;

testClass.MutableName = test1;

[test1 appendString:@"add 哈哈"];

打印结果:

2018-02-01 01:11:40.060493+0800 PropertyTest[1699:292294] test1:0x10af30150--test哈哈哈

2018-02-01 01:11:40.060721+0800 PropertyTest[1699:292294] testClass.Name:0x10af30130--test

2018-02-01 01:11:40.060951+0800 PropertyTest[1699:292294] testClass.MutableName:0xa000000747365744--test

    只有可变的NSMutableString test1的值改变了其他的值都没有改变。说明了copy是深拷贝

    将一个不可变的NSString test1赋值给Name、MutableName,再将test1的值改变,分别打印出来test1、Name、MutableName的内存地址。分析

NSString *test1 = @"test";

CopyClass *testClass = [[CopyClass alloc]init];

testClass.Name = test1;

testClass.MutableName = [[NSMutableString alloc]initWithString:test1];

test1 = @"test哈哈哈";

    打印结果:

2018-02-01 01:19:46.227394+0800 PropertyTest[1775:323385] test1:0x103701150--test哈哈哈

2018-02-01 01:19:46.227670+0800 PropertyTest[1775:323385] testClass.Name:0x103701130--test

2018-02-01 01:19:46.227900+0800 PropertyTest[1775:323385] testClass.MutableName:0xa000000747365744--test

     修改变量NSString Test1的值其他的值都没有改变。说明了copy是深拷贝

需要注意的是:

    用copy修饰一个NSMutableString的属性时,将一个可变或不可变的变量的值赋值给它时,用这个属性修改值的内容会崩溃。实例分析

// NSString *test1 = @"test";

NSMutableString *test1 = [[NSMutableString alloc]initWithString:@"test"];

CopyClass *testClass = [[CopyClass alloc]init];

testClass.MutableName = [[NSMutableString alloc]initWithString:test1];

[testClass.MutableName appendString:@"ahhaha"];

NSLog(@"test1:%p--%@",test1,test1);

NSLog(@"testClass.MutableName:%p--%@",testClass.MutableName,testClass.MutableName);

崩溃日志:

2018-02-01 01:26:12.133086+0800 PropertyTest[1860:346864] -[NSTaggedPointerString appendString:]: unrecognized selector sent to instance 0xa000000747365744

2018-02-01 01:26:12.137246+0800 PropertyTest[1860:346864] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[NSTaggedPointerString appendString:]: unrecognized selector sent to instance 0xa000000747365744'

    分析:reason: '-[NSTaggedPointerString appendString:]: unrecognized selector sent to instance   像NSTaggedPointerString这个实例发送appendString消息的时候没有找到这个方法,出现的崩溃。

    原因是copy修饰的属性最终的值都是不可变的类型,testClass.MutableName = [[NSMutableString alloc]initWithString:test1]; 这行代码最终生成的值是NSString类型。用NSString的实例去调用appendString出现的崩溃。  

    结论,用copy修饰的属性,都是不可变的变量。

自定义对象的copy

    实例:

CopyClass *testClass = [[CopyClass alloc]init];

testClass.Name = @"test";

testClass.MutableName = [[NSMutableString alloc]initWithString:@"test1"];

CopyClass *copytest = [testClass copy];

    直接崩溃:崩溃日志

2018-02-01 01:37:52.146383+0800 PropertyTest[1982:395784] -[CopyClass copyWithZone:]: unrecognized selector sent to instance 0x600000421c40

2018-02-01 01:37:52.152804+0800 PropertyTest[1982:395784] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[CopyClass copyWithZone:]: unrecognized selector sent to instance 0x600000421c40'

    分析:实例当中没有找到 copyWithZone方法,在自定义的类里面实现这个方法

- (instancetype)copyWithZone:(NSZone *)zone {

CopyClass *copyModel = [[CopyClass allocWithZone:zone] init];

copyModel.Name = self.Name;

copyModel.MutableName = self.MutableName; return copyModel;

}

    再次运行完美。

总结

    属性copy关键字是深拷贝;

    注意:1.这个属性被赋值的时候,都会转换成不可变的类型。

                2.拷贝对象的时候需要重写copyWithZone:方法。

发散

    copyWithZone   是什么?

    挖坑 

上一篇下一篇

猜你喜欢

热点阅读