NSCopying

定义:
NSCoping协议声明了一种提供对象副本的方法。复制的确切含义类类而异,但是复制必须是功能独立的对象,其值与复制时的原始值相同。使用NSCopying生成的副本由发送者隐式保留,发送者负责释放它。
NSCoping声明了一个方法copyWithZone:,但通常是用便捷方法“copy”。copy方法是为所有的NSObject定义的,只需要调用copyWithZone,在默认的zone。
使用NSCopying
NSCoping经常用于复制值对象--表示属性的对象。C类型变量通常可以替换值对象,但是值对象具有封装方便使用程序以进行常见操作的优点。例如,使用NSString对象而不是字符指针,因为他们封装了编码和存储。尽管有NSString功能,但NSString扮演的角色与角色指针所起的作用相似。
当值对象作为方法参数传递或从方法返回时,通常使用副本而不是对象本身。例如,考虑以下方法将字符串分配给对象的name的实例变量。

-(void)setName:(NSString *)aName {
      name = [aName copy];
}

存储aName的副本会产生一个独立于原始对象但具有相同内容的对象。后续对副本的更改不会影响原件,并且对原件的更改不会影响副本。类似的,通常返回实例变量的副本而不是实例变量本身。例如,此方法返回实例变量的副本:

- (NSString *)name {
      return [name copy];
}

声明NSCoping
创建副本有两种基本方法,您可以使用alloc和init...,或者可以使用NSCopyObject()。要选择一个适合你的类的方法,您需要考虑以下问题:

"What kind of copying-deep or shallow-does your class need?"

"Does your class's superclass implement NSCopying?"

"Are you familiar with the implementations of your class's superclasses?"

你的类需要什么样的复制 深还是浅?
通常,复制对象涉及创建新的实例并使用原始对象中的值对其进行初始化。复制非指针实例变量(如布尔值,整数和浮点)的值非常简单。复制指针实例变量时有两种方法:
浅拷贝:将指针值从原始对象复制到副本中。因此,原始和副本共享引用数据。
深拷贝:复制指针引用的数据并将其分配给副本的实例变量。
实例变量的set方法的实现应该反映您需要使用的复制类型。
如果相应的set方法复制新值,则应该深度复制实例变量,如下所示:

-(void)seMyVariable:(id)newValue {
      myVariable = [newValue copy];
}

如果相应的set方法保留新值,则应该浅复制实例变量:

- (void)setMyVariable:(id)newValue{
      myVariable = [newValue retain];
}

类似的,如果实例变量的set方法只是将新值分配给实例变量而不复制或保留它,则应如下所示:

- (void)setMyVariable:(id)newValue{
      myVariable = newValue;
}

要生成一个真正独立于原始对象的副本,必须深度复制整个对象。必须复制每个实例变量。如果实例变量本身具有实例变量,那么他们也必须重复,依次类推。在许多情况下,混合方法更有用。可以被认为是数据容器的指针的实例变量通常被深深地复制,而更为复杂的实例变量(如委托)被轻微复制。
例如,Product类采用NSCopying。Product实例具有此接口中声明的名称,价格和委托。

@interface Product: NSObject <NSCopying>
{
      NSString *productName;
      float price;
      id delegate;
}
@end

复制Product实例会生成productName的深层副本(深拷贝,复制指针引用的数据),因为它表示平面数据值。另一方面,委托实例变量是一个更复杂的对象,能够为两个产品正常运行。因此副本和原始对象应该共享委托。
productName的指针值不同,说明原始对象和拷贝对象都有自己的productName字符串对象。委托的指针值相同(浅拷贝,指针复制),表示两个产品对象与其委托共享同一对象。

Does your class's superclass implement NSCopying?
如果父类中没有实现NSCoping,那么你的类实现必须复制它集成的实例变量以及类中声明的变量。通常最安全的方法是使用alloc,init...和set方法。另一方面,如果你的类继承NSCopying行为,则其实现只需要复制你的类中声明的实例变量。他调用父类的实现来复制继承的实例变量。

Are you familiar with the implementations of your class's superclasses?
如果您的类继承了NSCopying行为,那么如何处理copyWithZone中的新实例变量:取决于您对父类的实现的熟悉程度。 基本上有两种方法可以使用alloc和init ...或使用函数NSCopyObject()来创建对象的副本。 如果父类使用或可能使用过NSCopyObject(),则必须以不同的方式处理实例变量。

使用alloc,init ...方法
如果类不继承NSCoping,则应实现copyWithZone:,使用alloc,init...和set方法。例如,对于上述Product类的copyWithZone:的实现可能以下列方式实现:

- (id)copyWithZone:(NSZone *)zone {
      Product *copy = [ [Product alloc] initWithProductName:[self productName] price:[self price] ];
      [copy setDelegate:[self delegate] ];
      return copy;
}

因为与继承的实例变量相关联的实现细节被封装在父类中,所以通常使用alloc,init...方法实现NSCopying更好。这样做使用set方法中实现的策略来确定实例变量所需的复制类型。

使用 NSCopyObject()
当一个类继承NSCopying行为时,必须考虑父类的实现使用NSCopyObject()的可能性。NSCopyObject()通过复制实例变量值而不是他们指向的数据来创建对象的精确浅复制。例如,NSCell的copyWithZone:的实现可以通过以下方式定义。

-(id)copyWithZone:(NSZone *)zone {
      NSCell *cellCopy = NSCopyObject(self, 0, zone);
      /* 假设其他初始化在这里*/
      cellCopy->image = nil;
      [cellCopy setImage:[self image] ];
      return cellCopy;
}

在上面的实现中,NSCopyObject()创建原始cell的精确浅复制。这种行为适用于复制非指针的实例变量或指向浅层复制的非保留数据的指针。保留对象的指针实例变量需要额外的处理。
在上面的copyWithZone: 示例中,image是指向保留对象的指针。保留图像的策略反应在setImage的存取方法中。

- (void)setImage:(NSImage *)anImage {
      image = [anImage retain];
}

如果copyWithZone的上述实现:在调用setImage:之前没有将副本的图像实例变量显式设置为nil,则副本引用的图像和原始图像将被释放而没有相应的保留。
即使图像指向正确的对象,它在概念上也是未初始化的。与使用alloc和init...创建的实例变量不同,这些未初始化的变量不是零值。在使用它们之前应该为这些变量显式分配初始值。在这种情况下,cellCopy的图像实例变量设置为nil,然后使用setImage:方法设置它。
NSCopyObject()的效果扩展到子类的实现。 例如,NSSliderCell的实现可以通过以下方式复制新的titleCell实例变量。

- (id)copyWithZone:(NSZone *)zone {
      NSSliderCell *cellCopy = [super copyWithZone:zone];
      
      cellCopy->titleCell = nil;
      [cellCopy setTitleCell:[self titleCell] ];
      return cellCopy;
}

调用父类的copyWithZone:方法来复制继承的实例变量。 当您调用父类的copyWithZone:方法时,如果父类实现有可能使用NSCopyObject(),则假定新的对象实例变量未初始化。 在使用它们之前,明确地为它们分配一个值。 在此示例中,在调用setTitleCell:之前,titleCell显式设置为nil。
使用NSCopyObject()时,对象的保留计数的实现是另一个考虑因素。 如果对象将其保留计数存储在实例变量中,则copyWithZone:的实现必须正确初始化副本的保留计数。
第一个对象表示内存中的Product实例。 refCount中的值表示实例已保留三次。 第二个对象是使用NSCopyObject()生成的Product实例的副本。 其refCount值与原始值匹配。 第三个对象表示从copyWithZone返回的副本:在正确初始化refCount之后。 在copyWithZone:使用NSCopyObject()创建副本后,它将值1分配给refCount实例变量。 copyWithZone的发件人:隐式保留副本并负责释放它。

NSCopying and Immutable Classes

当“不可变与可变”这一概念适用于某个对象时,NSCopying会生成不可变副本,无论原始副本是否为不可变。
不可变类可以非常有效地实现NSCopying。 由于不可变对象不会更改,因此无需复制它们。 相反,可以实施NSCopying以保留原始内容。 例如,copyWithZone:对于不可变的字符串类,可以通过以下方式实现。

- (id)copyWithZone:(NSZone *)zone {
      return [self retain];
}

总结

在不继承copyWithZone的类中使用alloc和init ...实现NSCopying:
通过调用父类的copyWithZone实现NSCopying:继承NSCopying行为时。 如果父类实现可能使用NSCopyObject(),则对保留对象的指针实例变量进行显式赋值。
当类及其内容不可变时,通过保留原始而不是创建新副本来实现NSCopying。

Instance Methods

copyWithZone:

  • (id)copyWithZone:(NSZone *)zone
    返回一个新实例,它是接收者的副本。 新实例的内存是从zone分配的,可能是NULL。 如果zone为NULL,则从默认区域分配新实例,该区域从NSDefaultMallocZone()返回。 返回的对象由发件人隐式保留,发件人负责释放它。 如果考虑“immutable vs. mutable”适用于接收对象,则返回的副本是不可变的; 否则,副本的确切性质由班级决定。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,357评论 8 265
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 从小学开始,我学习认真努力,成绩在班级还算优秀,为人低调,做事踏实,我就是邻居和老师眼中的“别人家的孩子”。遇到事...
    春天的蒲公英522阅读 203评论 0 0
  • 薛淮只是回来取合同,新剧本的合同。小远临走时忘记拿合同了,本来可以让剧组再打一份合同的,但他执意的想回来一趟。这么...
    老街木阅读 306评论 0 3
  • 骑车30分钟,上课上到9.40,骑了一会,感觉好累啊,哎,继续加油,感觉练习的不够,在公司就不要再看其他书了,好好...
    SPP164810阅读 52评论 0 0