ios底层原理程序员

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

2018-08-12  本文已影响0人  Qing学

我们知道系统的对象(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协议。不然并不能完全的完成对象拷贝。

上一篇下一篇

猜你喜欢

热点阅读