在Objective-c中, 某个类遵守了NSCopying协议就代表这个类支持[obj copy]操作。在没有实现NSCopying协议的情况下调用对象的copy函数则会出现异常。NSCopying协议只有一个函数,即copyWithZone, 声明如下 :
@protocol NSCopying - (id)copyWithZone:(NSZone *)zone; @endzone参数是新对象将分配到的内存区,一般不用修改。
遵守NSCopying协议则需要在@implementation中实现copyWithZone函数。
// Person @interface Person : NSObject <NSCopying> @property (nonatomic, copy) NSString* name; @property int age ; -(id) initWithName:(NSString*) pName andAge:(int) iAge ; @end
Person类的实现。
// 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
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