一、深淺拷貝
1、 什么是深拷貝?
深拷貝是對(duì)內(nèi)容的拷貝,復(fù)制內(nèi)容,同時(shí)有新的指針指向該內(nèi)存區(qū)域;
2、什么是淺拷貝?
淺拷貝只做指針復(fù)制,兩個(gè)指針指向同一處內(nèi)存空間;
3、對(duì)于拷貝,詳細(xì)來說有四方面內(nèi)容:
1、不可變對(duì)象的拷貝操作:immutableObjc -> copy
2、不可變對(duì)象的可變拷貝操作:immutableObjc -> mutableCopy
3、可變對(duì)象的拷貝操作:mutableObjc -> copy
4、可變對(duì)象的可變拷貝操作:mutableObjc -> mutableCopy
二、非集合對(duì)象的深淺拷貝(NSString、NSNumber...)
1、不可變對(duì)象的 copy 和 mutableCopy
NSString *string = @"string";
NSString *copyString = [string copy]; // 沒有產(chǎn)生新對(duì)象
NSMutableString *mutCopyString = [string mutableCopy]; // 產(chǎn)生新對(duì)象
NSLog(@"\n string = %p\n copyString = %p\n mutCopyString = %p", string, copyString, mutCopyString);
打印結(jié)果:
結(jié)論:對(duì)不可變對(duì)象 copy 是淺拷貝,mutablecopy 是深拷貝;
2、可變對(duì)象的 copy 和 mutableCopy
NSMutableString *string = [NSMutableString stringWithString:@"mutString"];
NSString *copyString = [string copy]; // 產(chǎn)生新對(duì)象
NSMutableString *mutCopyString = [string mutableCopy]; // 產(chǎn)生新對(duì)象
NSLog(@"\n string = %p\n copyString = %p\n mutCopyString = %p", string, copyString, mutCopyString);
打印結(jié)果:
結(jié)論:對(duì)可變對(duì)象 copy 和 mutablecopy 都是深拷貝;
3、非集合對(duì)象的深淺拷貝:
- immutableObjc -> copy 淺拷貝
- immutableObjc -> mutableCopy 深拷貝
- mutableObjc -> copy 深拷貝
- mutableObjc -> mutableCopy 深拷貝
三、集合對(duì)象的深淺拷貝(NSArray、NSSet、NSDictionary)
1、集合的淺復(fù)制 (shallow copy)
集合的淺復(fù)制有非常多種方法。當(dāng)你進(jìn)行淺復(fù)制時(shí),會(huì)向原始的集合發(fā)送retain消息,引用計(jì)數(shù)加1,同時(shí)指針被拷貝到新的集合。
現(xiàn)在讓我們看一些淺復(fù)制的例子:
NSArray *shallowCopyArray = [someArray copyWithZone:nil];
NSSet *shallowCopySet = [NSSet mutableCopyWithZone:nil];
NSDictionary *shallowCopyDict = [[NSDictionary alloc] initWithDictionary:someDictionary copyItems:NO];
2、集合的深復(fù)制 (deep copy)
方式一:用 initWithArray:copyItems: 將第二個(gè)參數(shù)設(shè)置為YES
NSDictionary *shallowCopyDict = [[NSDictionary alloc] initWithDictionary:someDictionary copyItems:YES];
如果使用這種方法深復(fù)制,集合里的每個(gè)對(duì)象都會(huì)收到 copyWithZone: 消息。如果集合里的對(duì)象遵循 NSCopying 協(xié)議,那么對(duì)象就會(huì)被深復(fù)制到新的集合。如果對(duì)象沒有遵循 NSCopying 協(xié)議,而嘗試用這種方法進(jìn)行深復(fù)制,會(huì)在運(yùn)行時(shí)出錯(cuò)。copyWithZone: 這種拷貝方式只能夠提供一層內(nèi)存拷貝(one-level-deep copy),而非真正的深復(fù)制。
方式二:將集合進(jìn)行歸檔(archive),然后解檔(unarchive)
NSArray *trueDeepCopyArray = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:oldArray]];
3、集合的單層深復(fù)制 (one-level-deep copy)
什么是單層深復(fù)制呢?如果在多層數(shù)組中,對(duì)第一層進(jìn)行內(nèi)容拷貝,其它層進(jìn)行指針拷貝,這種情況,即為單層深復(fù)制。蘋果的官方文檔
蘋果認(rèn)為這種復(fù)制不是真正的深復(fù)制,而是將其稱為單層深復(fù)制(one-level-deep copy)。因此,有人對(duì)淺復(fù)制、單層深復(fù)制、深復(fù)制做了概念區(qū)分。
- 淺復(fù)制(shallow copy):在淺復(fù)制操作時(shí),對(duì)于被復(fù)制對(duì)象的每一層都是指針復(fù)制。
- 單層深復(fù)制(one-level-deep copy):在深復(fù)制操作時(shí),對(duì)于被復(fù)制對(duì)象,至少有一層是深復(fù)制。
- 完全復(fù)制(real-deep copy):在完全復(fù)制操作時(shí),對(duì)于被復(fù)制對(duì)象的每一層都是對(duì)象復(fù)制。
1、不可變對(duì)象的 copy 和 mutableCopy
NSArray *array = @[@[@"a", @"b"], @[@"c", @"d"]];
NSArray *copyArray = [array copy];
NSMutableArray *mutCopyArray = [array mutableCopy];
NSLog(@"\narray - %p\ncopy - %p\nmutCopy - %p", array, copyArray, mutCopyArray);
打印結(jié)果:
結(jié)論:
- 查看內(nèi)容,可以看到 copyArray 和 array 的地址是一樣的,而 mutCopyArray 和 array 的地址是不同的。說明 copy 操作進(jìn)行了指針拷貝,mutableCopy 進(jìn)行了內(nèi)容拷貝。
- 需要強(qiáng)調(diào)的是:此處的內(nèi)容拷貝,僅僅是拷貝 array 這個(gè)對(duì)象,array 集合內(nèi)部的元素仍然是指針拷貝。這和上面的非集合 immutable 對(duì)象的拷貝還是挺相似的,
2、可變對(duì)象的 copy 和 mutableCopy
NSMutableArray *array = [NSMutableArray arrayWithObjects:[NSMutableString stringWithString:@"a"], @"b", @"c", nil];
NSArray *copyArray = [array copy];
NSMutableArray *mutCopyArray = [array mutableCopy];
NSLog(@"\narray - %p\ncopy - %p\nmutCopy - %p", array, copyArray, mutCopyArray);
查看結(jié)果:
結(jié)論:
- 根據(jù)打印結(jié)果可以看到 copyArray、mutCopyArray 和array 的內(nèi)存地址都不一樣,說明 copyArray、 mutCopyArray 都對(duì) array 進(jìn)行了內(nèi)容拷貝。
- 在集合類對(duì)象中,對(duì) immutable 對(duì)象進(jìn)行 copy,是指針復(fù)制,mutableCopy 是內(nèi)容復(fù)制;對(duì) mutable 對(duì)象進(jìn)行 copy 和 mutableCopy 都是內(nèi)容復(fù)制。但是:集合對(duì)象的內(nèi)容復(fù)制僅限于對(duì)象本身,對(duì)象元素仍然是指針復(fù)制。
4、集合對(duì)象的深淺拷貝
- [immutableObject copy] // 淺復(fù)制
- [immutableObject mutableCopy] //單層深復(fù)制
- [mutableObject copy] //單層深復(fù)制
- [mutableObject mutableCopy] //單層深復(fù)制
四、自定義對(duì)象
需要遵守<NSCopying>協(xié)議,并實(shí)現(xiàn)協(xié)議方法
CHIPerson *person = [[CHIPerson alloc] init];
CHIPerson *copyPerson = [person copy];
NSLog(@"\n person = %p\n copyPerson = %p\n", person, copyPerson);
- (id)copyWithZone:(NSZone *)zone
{
CHIPerson *person = [[self class] allocWithZone:zone];
return person;
}
打印結(jié)果:
結(jié)論:自定義對(duì)象的拷貝操作是深拷貝
本文參考鏈接: