copy关键字主要有两个要注意的地方,一个是何时使用copy关键字声明;第二个问题是深拷贝与浅拷贝的问题。
什么时候使用copy关键字
- 用于修饰block
- 在MRC时代,block内部的代码块是在栈区的、使用copy关键字可以把它放在堆区。
- 在ARC中,使用copy和strong效果相同。
- NSString、NSArray、NSDictionary
由于父类指针可以指向子类对象(NSString *可以指向NSMutableString对象),所以为了保护属性变量不会被修改,所以使用copy修饰。举例如下:
@interface ViewController ()
@property (nonatomic,strong,readwrite) NSString * strongedStr;
@property (nonatomic,copy,readwrite) NSString * copyedStr;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
NSMutableString * mutableStr = [[NSMutableString alloc]initWithString:@"123123"];
self.strongedStr = mutableStr;
self.copyedStr = mutableStr;
NSLog(@"原对象改变前的strongedStr---%@",self.strongedStr);
NSLog(@"原对象改变前的copyedStr---%@",self.copyedStr);
[mutableStr appendString:@"lalala"];
NSLog(@"原对象改变后的strongedStr---%@",self.strongedStr);
NSLog(@"原对象改变后的copyedStr---%@",self.copyedStr);
}
运行结果如下:
原对象改变前的strongedStr---123123
原对象改变前的copyedStr---123123
原对象改变后的strongedStr---123123lalala
原对象改变后的copyedStr---123123
可以看到,使用copy声明的对象可以保持当初被赋值的样子,即在内部拷贝了内容而不单单是指针。
深拷贝与浅拷贝
- 浅拷贝
- 源对象与拷贝对象在内存中是一份
- 源对象引用计数+1
- 拷贝了一份指向源对象的指针,并未产生新的对象
- 深拷贝
- 源对象与副本对象是两个不同的对象
- 源对象引用计数不变,拷贝对象计数器为1
- 本质是产生了新的对象
- copy与mutableCopy
不论源对象是否可变:- copy复制出的对象都是不可变的对象
- mutableCopy复制出的对象都是可变对象
- 基本类型不允许copy
- 自定义类型
- 自定义的类型要实现对应的NSCopying/NSMutableCopying协议:
-(id)copyWithZone:(NSZone *)zone {
PersonModel *model = [[[self class] allocWithZone:zone] init];
model.firstName = self.firstName;
model.lastName = self.lastName;
//未公开的成员--将其拷贝成不可变
model->_friends = [[NSMutableSet alloc] initWithSet:_friends copyItems:YES];
}
- (id)mutableCopyWithZone:(NSZone *)zone{
PersonModel *model = [[[self class] allocWithZone:zone] init];
model.firstName = self.firstName;
model.lastName = self.lastName;
//未公开的成员--将其拷贝成可变
model->_friends = [_friends mutableCopy];
return model;
}
- 集合对象
对于集合对象,要想实现深拷贝,需要自己手动地对内部元素一一拷贝。