深拷贝就是把内容拷贝一份产生一份新的对象,新对象计数器为1,源对象计数器不变。
而浅拷贝是指针拷贝,把地址给你,你和我指向同一个对象,源对象计数器加一,源对象和副本的计数器相同。
我们知道在OC中的拷贝函数有copy和mutablecopy,只要你调用了copy不管是NSString, NSDictionary还是NSArray还是NSMutableString还是NSMutableDictionary,还是NSMutableArray都是copy出来是不可变的副本。
当你调用mutablecopy产生的都是可变的副本。可变对象和不可变对象进行mutablecopy都是产生了一个新的对象,计数器为1,原来的对象计数器不变。而如果是copy而且是不可变对象进行copy的话则是浅拷贝没有产生一个新的对象相当于retain操作了,因为OC设置copy语法本身就是为了改变副本而不去影响源对象,所以为了性能着想既然copy出来的都不可变了,自然也就不会产生一个新的对象了。
可变对象进行copy就是深拷贝因为可变对象变成了不可变对象了,结构上发生了改变。总结来说就是只有不可变对象调用copy为浅拷贝其他都是深拷贝。
@property属性的copy代表的是set方法会release旧对象,copy一个产生新对象其实就是把retain的set方法中最后的retain改成了copy。所以其实就是这个意思.此处title是@property(copy,nonatomic)NSString * title;在manger这个类中
NSString *str1=[NSString stringWithFormat:@"hahha"];
Manger *manger1=[[Manger alloc]init];
manger1.title=str1;
NSLog(@"%i",manger1.title==str1);//输出是1,是浅拷贝
NSMutableString *str1=[NSMutableString stringWithFormat:@"hahha"];
Manger *manger1=[[Manger alloc]init];
manger1.title=str1;
NSLog(@"%i",manger1.title==str1);//输出为0,是深拷贝
NSString一般都用copy,不过还是得看需求。
如果我们想让一个类的对象进行拷贝操作就要让这个类要实现NSCopying或者NSMutableCopying协议。然后我们还要在类的.m文件中实现-(id)copyWithZone:(NSZone *)zone方法,其中zone指向新的一块存储空间,是系统已经给我们分配的内存。具体实现如下。
-(id)copyWithZone:(NSZone *)zone
{
Manger *copy=[[[self class] allocWithZone:zone]init];//这里就是用系统自己给分配的内存,此处最好用self class 来代替类名
copy.title=self.title;
return copy
}
总结:可变对象的copy和mutableCopy方法都是深拷贝,而且都是是单层拷贝也就是说我有个数组a和数组b,我让数组b = [a copy]或者 b = [a mutableCpoy],虽然b的地址和a的地址不同,但是其里面的数组元素的地址和原数组里面的对象的地址还是相同的,也就是说数组内容还是指向原数组里面的元素的。
不可变对象的copy方法是浅拷贝,mutableCopy的方法是深拷贝
copy方法返回的对象都是不可变对象
如果我们想实现数组中的对象也可以进行深拷贝的话,我们可以这么做,其实就是利用了归档的方式
- (void) deplyFullCopy
{
NSMutableArray *arrayM1 = [[NSMutableArray alloc] init];
NSMutableString *mstr1 = [[NSMutableString alloc]initWithString:@"哈哈"];
NSMutableString *mstr2 = [[NSMutableString alloc]initWithString:@"呵呵"];
[arrayM1 addObject:mstr1];
[arrayM1 addObject:mstr2];
NSData *data = [NSKeyedArchiver archivedDataWithRootObject:arrayM1];
NSArray *arrayM2 = [NSKeyedUnarchiver unarchiveTopLevelObjectWithData:data error:nil];
NSLog(@"marry1:%p --- %@ \r\n",arrayM1,arrayM1);
NSLog(@"marry2:%p --- %@ \r\n",arrayM2,arrayM2);
NSLog(@"数组元素地址:value1:%p --- value2:%p \r\n",arrayM1[0],arrayM1[1]);
NSLog(@"数组元素地址:value1:%p --- value2:%p \r\n",arrayM2[0],arrayM2[1]);
}