只复制指针成为浅拷贝,复制具有新的内存空间的对象称为深拷贝。
简单的说,浅拷贝是多个指针指向同一片内存,每一个指针对这片内存的操作,都会改变内存中的数据。深拷贝是将原来内存区域中的信息拷贝到一片新的内存中,每个指针指向一个独立的内存区域,只是这些内存中的数据(或者说对象)是完全一样的,某个内存中的改变并不会影响其他的内存。深拷贝和浅拷贝的本质指针指向的地址是否相同,地址相同,就是浅拷贝,地址不同就是深拷贝。
一般来说,对于遵守了NSCopying的类, 对象的copy是浅拷贝,mutablecopy是深拷贝。
例如:
NSString *string = @”hello";
NSString *stringCopy = [string copy];
NSMutableString *stringDCopy = [string mutableCopy];
[stringMCopy appendString:@"world"];
string和stringCopy指向的是同一块内存,是浅拷贝。而stringMCopy系统为其分配了新内存,是两个独立的字符串内容是一样的,是深拷贝。
对于还未遵守NSCopying或者NSMutableCopying的类,可以手动遵守这个协议,实现复制。在说明复制方法的自定义之前,需要说明:
(1)cocoa中把动态分配的内存管理称之为区域(zone),然而在新的运行时系统中,不再使用区域;
(2)NSObject中的copy可以生成新的实例,但是实际的复制操作并不是copy完成的,而是由实例方法copyWithZone:完成的。给实例对象发送copy消息后,参数被赋值为NULL,并调用copyWithZone:。
鉴于以上的说明,要想实现实例的复制,就必须要自定义方法copyWithZone:。
// NScopying协议
@protocol NSCopying
-(id)copyWithZone:(NSZone *)zone;
@end
对于copyWithZone:方法进行总结:
(1)父类实现了copyWithZone:,可直接调用生成实例,然后将子类中添加的实例变量根据需要进行复制并带入;
(2)对于常数对象,定义copyWithZone:,并不一定生成新的对象;
简单例子说明:
Animal类包含了name,age,location三个属性,我们假定都是中国的动物,location是一定的,在复制Animal类的实例时,就没有必要复制location,只需要共享指针就可以啦
#import "Animal.h"
@interface Animal ()<NSCopying>
{
NSString *name;
NSString *age;
NSString *location;
}
@end
接下来定义copyWithZone:方法:
@implementation Animal
-(id)copyWithZone:(NSZone *)zone
{
// 这里一定要用self class,因为子类会继承这个方法,而子类返回的实例对象的类型,
// 应该为子类的类型,不应该使用父类的类名
Animal *tempAnimal = [[[self class]allocWithZone:zone]init];
if (tempAnimal) {
tempAnimal->name = [name copyWithZone:zone];
tempAnimal->age = [age copyWithZone:zone];
tempAnimal->location = [location copyWithZone:zone];
}
return tempAnimal;
}
@end
这个地方一定注意不能使用类型进行实例的创建,而要用[self class]。
假如有一个类People继承了类Animal,并且增加了一个属性thought,由于父类遵守了协议NSCopying,People中可以不指定协议;
#import "People.h"
@interface People():Animal
{
NSString *thought;
}
@end
在People中重新定义copyWithZone:
-(id)copyWithZone:(NSZone *)zone
{
// 这里一定要用self class,因为子类会继承这个方法,而子类返回的实例对象的类型,
// 应该为子类的类型,不应该使用父类的类名
People *tempPeople = [[[self class]allocWithZone:zone]init];
if (tempPeople) {
tempAnimal->thought = [thought copyWithZone:zone];
}
return tempPeople;
}
这个方法返回的是People的实例对象而不是Animal,这就是为什么要用[self class]而不用类名的原因。