在Objective-c中, 某个类遵守了NSCopying协议就代表这个类支持[obj copy]操作。在没有实现NSCopying协议的情况下调用对象的copy函数则会出现异常。NSCopying协议只有一个函数,即copyWithZone, 声明如下 :
@protocol NSCopying
- (id)copyWithZone:(NSZone *)zone;
@end
zone参数是新对象将分配到的内存区,一般不用修改。
遵守NSCopying协议则需要在@implementation中实现copyWithZone函数。
// Person
@interface Person : NSObject <NSCopying>
@property (nonatomic, copy) NSString* name;
@property int age ;
-(id) initWithName:(NSString*) pName andAge:(int) iAge ;
@endPerson类的实现。
// impl
@implementation Person
@synthesize name, age ;
//
-(id) initWithName:(NSString*) pName andAge:(int) iAge
{
self = [super init] ;
if ( self ) {
self.name = pName ;
self.age = iAge ;
}
return self ;
}
// 实现copyWithZone
-(id) copyWithZone:(NSZone *)zone
{
NSLog(@"Person copyWithZone, class : %@.", [self.class description]) ;
// 如果子类覆写该方法, 则[self class]的类型为子类型,如果直接使用
// [[Person allocWithZone:zone] init], 则子类覆写时会出错.
Person* per = [[[self class] allocWithZone:zone] init] ;
per.name = self.name ;
per.age = self.age ;
return per ;
}
@end
// impl
@implementation Person @synthesize name, age ;
//
-(id) initWithName:(NSString*) pName andAge:(int) iAge
{
self = [super init] ;
if ( self ) {
self.name = pName ;
self.age = iAge ;
}
return self ;
} // 实现copyWithZone
-(id) copyWithZone:(NSZone *)zone
{
NSLog(@"Person copyWithZone, class : %@.", [self.class description]) ;
// 如果子类覆写该方法, 则[self class]的类型为子类型,如果直接使用
// [[Person allocWithZone:zone] init], 则子类覆写时会出错.
Person* per = [[[self class] allocWithZone:zone] init] ;
per.name = self.name ;
per.age = self.age ;
return per ;
} @end
在copyWithZone函数中创建了一个对象,并且将属性挨个拷贝给新的对象。注意新对象per的创建方法是
[[[self class] allocWithZone:zone] init] ;
而不是
[[Person allocWithZone:zone] init] ;
这是因为如果有子类覆写了copyWithZone函数,那么在子类对象被拷贝时会调用Person类的copyWithZone函数, 如果硬编码类型为Person, 则创建的对象为Person类型, 而不是子类型,因此会出现异常。而[self class]则会获取正确的类型, 并且创建出正确的对象,避免异常出现。
例如Student类继承自Person类。
// subclass, Student
@interface Student : Person
// 班级名
@property (nonatomic, copy) NSString* className ; @end
// subclass, Student
@interface Student : Person
// 班级名
@property (nonatomic, copy) NSString* className ; @end
Student的实现,覆写copyWithZone函数。
// Student实现
@implementation Student // 覆写copyWithZone,
-(id) copyWithZone:(NSZone *)zone
{
// 1、先调用父类的copyWithZone函数
Student* per = [super copyWithZone:zone] ;
// 2、再设置className
per.className = self.className ;
NSLog(@"Student copyWithZone") ;
return per ;
}
@end