一说到拷贝,就不得不提浅拷贝和深拷贝。
何谓浅拷贝?何谓深拷贝?
往简单的说:
- 浅拷贝:拷贝地址。
- 深拷贝:拷贝内容。
用通俗的话来举例子:
- 浅拷贝:就是找锁匠配了把新钥匙,虽然钥匙是新的,开的却还是之前那扇门,进的还是之前那间屋。
- 深拷贝:不仅买了套户型一模一样的房子,还定制了一模一样的家具,购买了品牌与尺寸一模一样的家电,放置了一模一样的私人用品,甚至连摆放的位置,使用的痕迹都得一模一样。(这样的细节问题,甚至可以无限深入下去)
那如果仅仅是买了套户型一模一样的房子,至于家具家电私人用品等等,却依然用的原来的。这又算深拷贝还是浅拷贝呢?
苹果给它取了个名字one-level-deep copy
,译名单层深拷贝
copy 与 mutableCopy
在iOS里,想让一个类的对象通过调用copy
方法实现拷贝的话,并非手动实现copy
方法,而是需要遵循并实现NSCopying
协议。该协议只有一个方法:
@protocol NSCopying
- (id)copyWithZone:(nullable NSZone *)zone;
@end
类似的,与mutableCopy
对应的协议NSMutableCopying
:
@protocol NSMutableCopying
- (id)mutableCopyWithZone:(nullable NSZone *)zone;
@end
Foundation
框架中的类已遵循并实现协议NSCopying
或NSMutableCopying
。类名以NSMutable
开头的,称为可变类,其余的叫不可变类。可变类总是有一个不可变类与之对应(反之不然),比如NSString
与NSMutableString
,NSArray
与NSMutableArray
。copy
的结果总是不可变类,mutableCopy
的结果总是可变类。
示例代码
下面以代码的形式来探讨copy
与mutableCopy
在Foundation
框架中的区别。
NSString *originString = [NSString stringWithFormat:@"Test String %d", 1];
NSString *copyString = [originString copy];
NSMutableString *mutableString = [originString mutableCopy];
NSString *m2CopyString = [mutableString copy];
NSMutableString *m2MutableString = [mutableString mutableCopy];
NSLog(@"originString : %p", originString);
NSLog(@"copyString : %p", copyString);
NSLog(@"mutableString : %p", mutableString);
NSLog(@"m2CopyString : %p", m2CopyString);
NSLog(@"m2MutableString : %p", m2MutableString);
NSLog(@"************* 分割线 ***************");
NSArray *originArr = @[originString, mutableString];
NSArray *copyArr = [originArr copy];
NSMutableArray *mutableArr = [originArr mutableCopy];
NSArray *m2CopyArr = [mutableArr copy];
NSMutableArray *m2MutableArr = [mutableArr mutableCopy];
void (^showLogBlock)(NSArray *anArr) = ^(NSArray *anArr){
for (int i = 0; i < anArr.count; i++) {
NSLog(@"No.%d : %p", i, [anArr objectAtIndex:i]);
}
};
NSLog(@"originArr : %p", originArr);
showLogBlock(originArr);
NSLog(@"copyArr : %p", copyArr);
showLogBlock(copyArr);
NSLog(@"mutableArr : %p", mutableArr);
showLogBlock(mutableArr);
NSLog(@"m2CopyArr : %p", m2CopyArr);
showLogBlock(m2CopyArr);
NSLog(@"m2MutableArr : %p", m2MutableArr);
showLogBlock(m2MutableArr);
结果输出
originString : 0x60000002b4c0
copyString : 0x60000002b4c0
mutableString : 0x60000007b200
m2CopyString : 0x60000002b380
m2MutableString : 0x60000007b040
************* 分割线 ***************
originArr : 0x60800002b000
No.0 : 0x60000002b4c0
No.1 : 0x60000007b200
copyArr : 0x60800002b000
No.0 : 0x60000002b4c0
No.1 : 0x60000007b200
mutableArr : 0x608000045820
No.0 : 0x60000002b4c0
No.1 : 0x60000007b200
m2CopyArr : 0x60800002afe0
No.0 : 0x60000002b4c0
No.1 : 0x60000007b200
m2MutableArr : 0x608000045850
No.0 : 0x60000002b4c0
No.1 : 0x60000007b200
结论
非容器类 | copy | mutableCopy |
---|---|---|
不可变类 | 浅拷贝 | 深拷贝 |
可变类 | 深拷贝 | 深拷贝 |
容器类 | copy | mutableCopy |
---|---|---|
不可变类 | 浅拷贝 | 单层深拷贝 |
可变类 | 单层深拷贝 | 单层深拷贝 |
自定义类的copy与mutableCopy
Foundation
框架以外的类,需手动遵循并实现协议NSCopying
才能调用copy
方法。
- 若父类没有实现
NSCopying
:
- (id)copyWithZone:(NSZone *)zone{
CustomModel *model = [[[self class] allocWithZone:zone] init];
model.propertyObject = [_propertyObject copyWithZone:zone];
model.propertyValue = _propertyValue;
model->_variableObject = [_variableObject copyWithZone:zone];
model->_variableValue = _variableValue
return model;
}
- 若父类已实现
NSCopying
:
- (id)copyWithZone:(NSZone *)zone{
SubCustomModel *subModel = [super copyWithZone:zone];
......
return subModel;
}
至于手动遵循并实现NSMutableCopying
协议,方法同NSCopying
一样。