原型模式从一个对象在创建另一个对象,而不需知道任何创建细节。一般在初始化信息不变化的情况下,使用原型模式是最好的方法,即隐藏了对象创建的细节,对性能又大大提高。
在 iOS 开发中,体现原型模式的是 copy 和 mutableCopy。
copy 指的是复制对象,返回一个不可变的对象。
NSArray *arrA = @[@"1",@"2",@"3",@"4"];
id arrACopy = arrA.copy;
arrACopy[0] = @"2";//崩溃,因为copy返回的对象是不可变的
arrA = @[@"2",@"2",@"3",@"4"];//arrA 被重新赋值,地址变化
执行前两句之后打断点看 arrA 和 arrACopy 的地址,他们的地址是相同的。这是浅拷贝的特性,浅拷贝拷贝对象的地址,拷贝后,B对象与A对象的值相同,而地址也相同。说明这里是浅拷贝。
第三句会引起崩溃,因为想强行给 immutable 的对象设值。copy 返回的对象都是 immune 的。
mutableCopy指的是复制对象,返回一个可变的对象。
NSArray *arrA = @[@"1",@"2",@"3",@"4"];
id arrAMutableCopy = arrA.mutableCopy;
arrAMutableCopy[0] = @"2";//设值成功
执行前两句之后打断点,arrA 和 arrAMutableCopy 的值是相同的,地址不同。这里是深拷贝,深拷贝拷贝对象的具体内容,拷贝后,B对象与A对象的值相同,但地址不同。
要区分深浅拷贝,只需打印对象地址,然后比较地址是否相同。举个浅拷贝的例子:
//浅拷贝,执行 sectionNames addObject:后 sectionsArray[0] 一样也 addObject: 了
NSMutableArray *sectionNames = sectionsArray[0];
[sectionNames addObject:str];
自己建立的对象,想要使用 copy,需要遵守 NSCopying 协议,并且实现 copyWithZone: 方法。
- (id)copyWithZone:(NSZone *)zone {
Resume *resumeCopy = [[self class] allocWithZone:zone];
resumeCopy.name = [self.name copy];
resumeCopy.age = [self.age copy];
resumeCopy.sex = [self.sex copy];
resumeCopy.delegate = self.delegate;
return resumeCopy;
}
想要使用 mutableCopy,需要遵守 NSMutableCopying 协议,并且实现 mutableCopyWithZone: 方法。
实例地址:https://github.com/clairehu7/DesignPatterns(06Prototype)