引言
最近项目中遇到一些有关自定义对象的copy问题,今天在这里做一些记录。以便加深理解和记忆。
什么是深拷贝、浅拷贝?
- 深拷贝:指针赋值,且重新开辟内存,内容重新拷贝一份。
- 浅拷贝:指针赋值,指针指向的内容是同一个地址,内容的引用计数+1。
注意在浅拷贝的时候如果指针指向的内存区域被销毁,指向这片内存的指针都要重新定义不然会成为野指针。
1.非集合对象的copy与mutableCopy
1.1 NSString
NSString *string1 = @"str1";
//copy返回的是不可变对象,str2不能被修改,因此会发生崩溃
NSString *string2 = [string1 copy];
NSLog(@"string1: %p %p,string2: %p %p",string1,&string1,string2,&string2);
NSMutableString *string3 = [string1 mutableCopy];
NSLog(@"string1: %p %p,string3: %p %p",string1,&string1,string3,&string3);
2017-08-03 20:26:26.450 TableViewTest[77698:6624676] string1: 0x10092e0f8 0x7fff5f2d5980,string2: 0x10092e0f8 0x7fff5f2d5978
2017-08-03 20:26:26.450 TableViewTest[77698:6624676] string1: 0x10092e0f8 0x7fff5f2d5980,string3: 0x608000077b00 0x7fff5f2d5970
从打印的内容可以看的出来string1、string2内容的地址相同,而string1、string2指针的地址不同。string1、string3内容的地址和指针地址都不相同。可知NSString的copy为浅拷贝,mutablecopy为深拷贝
1.2 NSMutableString
NSMutableString *mstr1 = [NSMutableString stringWithString:@"test002"];
NSMutableString *mstr2 = [mstr1 copy];
NSLog(@"mstr1: %p %p,mstr2: %p %p",mstr1,&mstr1,mstr2,&mstr2);
//copy返回的是不可变对象,mstr2不能被修改,因此会发生崩溃
//[mstr2 appendString:@"test"];
NSMutableString *mstr3 = [mstr1 mutableCopy];
//[mstr3 appendString:@"modify"];
NSLog(@"mstr1: %p %p,mstr3: %p %p",mstr1,&mstr1,mstr3,&mstr3);
2017-08-03 20:49:28.896 TableViewTest[78081:6667312] mstr1: 0x6000000767c0 0x7fff51c74980,mstr2: 0xa323030747365747 0x7fff51c74978
2017-08-03 20:49:28.896 TableViewTest[78081:6667312] mstr1: 0x6000000767c0 0x7fff51c74980,mstr3: 0x600000076800 0x7fff51c74970
从打印的内容可以看的出来mstr1、mstr2、mstr3内容地址和指针地址都不相同。所以NSMutableString的copy和mutableCopy都是深拷贝。且copy返回的对象是不可变对象
2. 集合对象
2.1 不可变对象NSArray
NSArray *arry1 = [[NSArray alloc] initWithObjects:@"value1", @"value2",nil];
NSArray *arry2 = [arry1 copy];
NSArray *arry3 = [arry1 mutableCopy];
NSLog(@"arry1: %p %p,arry2: %p %p",arry1,&arry1,arry2,&arry2);
NSLog(@"arry1: %p %p,arry3: %p %p",arry1,&arry1,arry3,&arry3);
2017-08-03 20:58:29.940 TableViewTest[78209:6684329] arry1: 0x608000223a00 0x7fff5f16f980,arry2: 0x608000223a00 0x7fff5f16f978
2017-08-03 20:58:29.940 TableViewTest[78209:6684329] arry1: 0x608000223a00 0x7fff5f16f980,arry3: 0x608000245280 0x7fff5f16f970
从打印的内容可以看的出来arry1、arry2内容的地址相同,而arry1、arry2指针的地址不同。arry1、arry3内容的地址和指针地址都不相同。可知NSString的copy为浅拷贝,mutablecopy为深拷贝
2.2 可变对象NSMutableArray
NSMutableArray *arry1 = [[NSMutableArray alloc] initWithObjects:@"value1", @"value2",nil];
NSMutableArray *arry2 = [arry1 copy];
//copy返回的是不可变对象,marry2不能被修改,因此会崩溃
//[arry2 addObject:@"value3"];
NSMutableArray *arry3 = [arry1 mutableCopy];
NSLog(@"arry1: %p %p,arry2: %p %p",arry1,&arry1,arry2,&arry2);
NSLog(@"arry1: %p %p,arry3: %p %p",arry1,&arry1,arry3,&arry3);
从打印的内容可以看的出来arry1、arry2、arry3内容地址和指针地址都不相同。所以NSMutableString的copy和mutableCopy都是深拷贝。且copy返回的对象是不可变对象