zoukankan      html  css  js  c++  java
  • 第22条:理解NSCopying协议

    如果想自定义类支持拷贝操作,那就要实现NSCopying协议(而不是复写copy方法)或 NSMutableCopying的协议。

    不可变版本的拷贝:

    NSCopying协议,该协议只有一个方法:

    -(id)copyWithZone:(NSZone*)zone; 

    // 以前开发程序时,会据此把内存分成不同的“区”(zone),而对象会创建在某个区里面。现在不用了,每个程序只有一个区:“默认区”(default zone)。

    例:

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

      Person *copy = [[[self class]allocWithZone:zone]initWithFirstName:_firstName andLastName:_lastName]; // 这里使用了"全能初始化方法"

      // copy->_friends = [_friends mutableCopy]; // 类中的数据结构可能并未在初始化方法或“全能初始化方法”中设置好,需要另行设置。

      // 这里的_friends 是.m中使用的实例变量。

      // 或

      // copy->_friends = [NSMutableSet alloc]initWithSet:_friends copyItems:YES]; // 若copyItem为YES,则该方法会向数组中的每个元素发送copy消息。用拷贝好的元素创建新的set。

      // 这两种方式说明:遵从NSCopying协议的对象不一定采用了深拷贝的方式来实现的。

      return copy;

    }

    可变版本的拷贝:

    NSMutableCopying的协议。也只定义了一个方法:

    -(id)mutableCopyWithZone:(NSZone*)zone;

    注意:

    在可变对象上调用copy方法会返回另外一个不可变类的实例(mutableCopy反之)。这样做是为了能在可变版本与不可变版本之间自由切换。

    不过也可能出现某个方法将一个NSMutableArray对象当作NSArray返回给你,而你在上面调用copy方法来复制它。由此以为拷贝后的对象是不可变的数组,但实际上它却是可变的(运行时特性),所以这就出问题了。

    可以查询类型信息(参见第14条)以判断待拷贝的实例是否可变。

    深拷贝(deep copy)和 浅拷贝(shallow copy)

    深拷贝:在拷贝对象自身(指针)时,将其底层数据也一并复制过去。

    浅拷贝:只拷贝对象自身(指针).

    Foundation框架中的所有collection类在默认情况下都执行浅拷贝,

    总结:

    如果想自定义类支持拷贝操作,则需实现NSCopying协议。

    如果自定义的对象分为可变版本和不可变版本,那么就要同时实现NSCopying与NSMuableCopying协议。

    复制对象时需决定采用浅拷贝还是深拷贝,一般情况下应该尽量执行浅拷贝。

    如果你所写的对象需要深拷贝,那么可考虑新增一个专门执行深拷贝的方法。

  • 相关阅读:
    网络安全法课程上线喽~
    网络安全法将在六一正式实施,我该如何继续爱你?
    安卓逆向入门(一)
    如何正确的使用Ubuntu以及安装常用的渗透工具集.
    Linux/CentOS各种服务框架的搭建完整流程
    【Anroid界面实现】WindowManager类使用具体解释——用户首次打开APP的使用教学蒙板效果实现
    2014年辛星jquery解读第三节 Ajax
    迅为4412开发板Linux驱动教程/硬件知识及原理图的使用
    点评国内各大在线app生成平台
    OpenGL ES2.0 基本编程
  • 原文地址:https://www.cnblogs.com/Pikdays/p/5735101.html
Copyright © 2011-2022 走看看