定义:
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”适用于接收对象,则返回的副本是不可变的; 否则,副本的确切性质由班级决定。