对于iOS中浅拷贝和浅拷贝,大家只是模糊的知道前者是复制了一份对象的地址,也就是同一个对象增加了一个指向它的指针,而后者是一个新的指针变量指向一个新的对象,这个新的对象是原对象的复制品,即对象属性完全相同只是位于不同的内存空间。但针对数组类型的对象时还需要更仔细理解一下。
数组类型的浅拷贝同样是开辟一个新的指针指向该数组,两个指针操作的是同一个数组;而深拷贝分为两种情况,一是拷贝数组中的对象地址至一个新的数组,新数组中对象和原数组中对象是相同的内存地址即相同的实例,二是拷贝数组中的对象至一个新的数组,新数组中对象和原数组中对象是不同的实例。
还是直接用代码输出拷贝的数组地址来解释吧。
首先创建数组arrayA,其中添加了两个对象person1和person2,并且输出arrayA的地址如图1所示。
NSMutableArray *arrayA = [[NSMutableArray alloc]initWithCapacity:0];
Person *person1 = [[Person alloc]initWithHeight:180 withAge:18 withName:@"Joe"];
Person *person2 = [[Person alloc]initWithHeight:150 withAge:15 withName:@"Kate"];
[arrayA addObject:person1];
[arrayA addObject:person2];
NSLog(@"arrayA:%p,%@",arrayA,arrayA);
1.原数组arrayA的内存地址和所含对象的内存地址
然后创建arrayB数组,用来拷贝arrayA,并且向arrayB中添加对象person3,输出arrayB和arrayA的地址如图2所示,两个数组完全一致,arrayB其实就是arrayA的浅拷贝。
NSMutableArray *arrayB = arrayA;
Person *person3 = [[Person alloc]initWithHeight:170 withAge:23 withName:@"Mike"];
[arrayB addObject:person3];
NSLog(@"arrayB:%p,%@",arrayB,arrayB);
NSLog(@"arrayA:%p,%@",arrayA,arrayA);
2.浅拷贝之arrayB和arrayA的内存地址
创建数组arrayA的备份数组arrayC,并且向arrayC中添加对象person4,输出arrayC的内存地址和数组中对象的地址如图3,可以发现arrayC和arrayA的地址不同,但是其中的对象地址却相同,也就是说arrayC数组中的对象和arrayA数组中对象是同一份,指向相同的内存地址。当对数组arrayA中的person1操作,那么数组arrayC中的person1也会相应的变化。
NSMutableArray *arrayC = [[NSMutableArray alloc]initWithArray:arrayA];
Person *person4 = [[Person alloc]initWithHeight:175 withAge:24 withName:@"赵子龙"];
[arrayC addObject:person4];
NSLog(@"arrayC:%p,%@",arrayC,arrayC);
3.深拷贝arrayC的内存地址和数组中对象地址
有时候,我们想两份数组分别操作时而互不影响其中的对象,也就是两个数组位于不同的内存地址,并且所含的对象属性完全相同但并不是同一个实例,那么就需要在深拷贝数组的同时深拷贝数组中的对象。代码如下,创建arrayD的同时拷贝arrayA中的所有对象,并且向arrayD中添加对象person5,输出arrayD的内存地址和所含对象的地址如图4所示。
NSMutableArray *arrayD = [[NSMutableArray alloc]initWithArray:arrayA copyItems:YES];
Person *person5 = [[Person alloc]initWithHeight:177 withAge:44 withName:@"赵云"];
[arrayD addObject:person5];
NSLog(@"arrayD:%p,%@",arrayD,arrayD);
4.arrayD的内存地址和所含对象的内存地址
arrayD的内存地址arrayA的内存地址不同,而且所含对象的地址也是不同的,这样在操作arrayD和arrayA的时候,两者是互不影响的。
备注,NSMutableArray *arrayD = [[NSMutableArray alloc]initWithArray:arrayA copyItems:YES];这句代码的使用需要数组中的对象遵循NSCoping和NSMutableCopying协议,也就是这里的Person类重载如下的两个方法,否则直接用上面的API会崩溃。
-(id)copyWithZone:(NSZone *)zone;
-(id)mutableCopyWithZone:(NSZone *)zone;
重载代码的示例如下:
-(id)copyWithZone:(NSZone *)zone
{
Person *A = [[self class] allocWithZone:zone];
A.age = self.age;
A.height = self.height;
A.name = self.name;
return A;
}
如有疑问欢迎指正!