谈谈NSCopy协议和OC自定义对象的拷贝

我们知道系统的对象(NSString,NSArray..)有拷贝方法,具体的可以分为copy和mutableCopy两种。但是对于我们自定义的对象是不能直接调用copy方法的。会直接崩溃报错如下

reason: '-[Person copyWithZone:]: unrecognized selector sent to instance 

想直接调用copy实现拷贝需要自定义的对象实现NSCopy协议。同时实现copyWithZone方法。
下面我们将通过代码实现自定义对象的copy并且对于对象及对象的属性的指针和存储地址进行分析。
假设创建自定义对象Person
Person.h

#import <Foundation/Foundation.h>
@class Man;

@interface Person : NSObject<NSCopying,NSMutableCopying>

@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) NSInteger age;
@property (nonatomic, strong) Man *man;

@end

可以看到又三个属性。字符串name,整形age,和自定义对象Man。接下来我们要在.m文件和Man类中实现copyWithZone方法。
Person.m

#import "Person.h"
#import "Man.h"

@implementation Person

- (id)copyWithZone:(NSZone *)zone{
    Person *xiaoM = [Person allocWithZone:zone];
    xiaoM.name = self.name;
    xiaoM.age = self.age;
    xiaoM.man = [self.man copy];
    return xiaoM;
}

@end

man.h的声明如下

#import <Foundation/Foundation.h>

@interface Man : NSObject<NSCopying>

@property (nonatomic, strong) NSString *name;

@end

man.m类的实现如下

#import "Man.h"

@implementation Man

- (id)copyWithZone:(NSZone *)zone{
    Man *man = [Man allocWithZone:zone];
    man.name = self.name;
    return man;
}

@end

接下来在程序中进行调用。分析copy情况

Person *personDefault = [[Person alloc]init];
    personDefault.name = @"qx";
    personDefault.age = 25;
    Man *baseMan = [[Man alloc]init];
    baseMan.name = @"2333";
    personDefault.man = baseMan;
    Person *personCopy = [personDefault copy];
    NSLog(@"oldPerson%@----CopyPerson%@",personDefault,personCopy);
    NSLog(@"personMan%@----copyMan%@",personDefault.man,personCopy.man);
    NSLog(@"personMan%@----CopyMan%@",personDefault.man.name,personCopy.man.name);
    personDefault.man.name = @"我是小明";
    NSLog(@"personMan%@----CopyMan%@",personDefault.man.name,personCopy.man.name);

调试器的输出如下
DesignCopy[52351:1845750] oldPerson<Person: 0x60000003de20>----CopyPerson<Person: 0x60000003dde0>
DesignCopy[52351:1845750] personMan<Man: 0x60000000ba50>----copyMan<Man: 0x60000000b9f0>
DesignCopy[52351:1845750] personMan2333----CopyMan2333
DesignCopy[52351:1845750] personMan我是小明----CopyMan2333
分析如下:
通过第一行的输出分析我们可以看到personDefault的存储地址跟copyPerson的存储地址不同。对于person的copy操作完成了person对象的深拷贝。
通过第二行的分析我们可以看到man的存储地址也发生了变化(因为Man也实现了Copy协议)在这里完成了深拷贝
通过第三行和第四行的分析我们可以看到。本来相同的字符串2333.在对personDefault的man对象进行name属性的修改后。拷贝的对象的内容并没有发生改变。所以确定对于对象里面的元素也完成了深拷贝。

--下面我们来分析下对于Man类不实现copy协议会怎样
如果把Man的copyWithZone方法注释掉。在person.m文件中直接赋值。修改后的代码如下
person.m

- (id)copyWithZone:(NSZone *)zone{
    Person *xiaoM = [Person allocWithZone:zone];
    xiaoM.name = self.name;
    xiaoM.age = self.age;
    xiaoM.man = self.man;
    return xiaoM;
}

man.m

#import "Man.h"

@implementation Man

//- (id)copyWithZone:(NSZone *)zone{
//    Man *man = [Man allocWithZone:zone];
//    man.name = self.name;
//    return man;
//}

@end

同样的输出结果变成了

2018-08-12 21:21:47.840126+0800 DesignCopy[52385:1850578] personMan2333----CopyMan2333
2018-08-12 21:21:47.840729+0800 DesignCopy[52385:1850578] personMan我是小明----CopyMan我是小明

可以看到对于man进行了浅拷贝。两个man对象的name属性还是指向了一块内存地址。
所以我们实现自定义对象的copy的拷贝时,对于属性里面的自定义对象也要实现NSCopy协议。不然并不能完全的完成对象拷贝。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,161评论 1 32
  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,688评论 8 265
  • 前言 不敢说覆盖OC中所有copy的知识点,但最起码是目前最全的最新的一篇关于 copy的技术文档了。后续发现有新...
    zyydeveloper阅读 3,451评论 4 35
  • 张红云 平顶山 坚持分享第263天( 原创总第431天 2018年2月9日 星期五) 有时候总觉得时间不够用...
    红云_杨柳清风阅读 378评论 0 0
  • 任何时候,不要高估你跟其他人的情谊。 任何时候,不要向朋友提出让他为难的请求。 不到万不得已,自己多苦多难受都好,...
    Klex阅读 249评论 0 2