美文网首页ios底层原理程序员
谈谈NSCopy协议和OC自定义对象的拷贝

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

作者: Qing学 | 来源:发表于2018-08-12 21:26 被阅读0次

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

    相关文章

      网友评论

        本文标题:谈谈NSCopy协议和OC自定义对象的拷贝

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