您的位置:首页 > 移动开发 > IOS开发

ios 深拷贝和浅拷贝

2016-03-30 10:25 363 查看
我们先明白这么一句话 : 赋值语句仅仅创建了对内存中同一对象的另一个引用。

下面我们来看一个例子

NSMutableArray *dataArray = [NSMutableArray arrayWithObjects:@"one",@"two",@"three",@"four", nil];

// 赋值语句仅仅创建了对内存中同一数组对象的另一个引用
NSMutableArray *array = dataArray;
[array removeObjectAtIndex:0];

for (NSString *item in dataArray) {

NSLog(@"dataArray:%@",item);
}
for (NSString *item in array) {

NSLog(@"array:%@",item);
}


输出结果: 两个输出结果一致: two,three,four

以上代码块 的输出结果很显然: 赋值语句 array = dataArray 仅仅是创建了对内存中统一数组对象的另一个引用。这两个数组中的元素都指向内存中同一个字符串。

上面要注意一个细节: 产生一个对象的可变副本并不要求被复制的对象本身是可变的。

一、浅拷贝(克隆影子,不同影子指向同一个人人)

浅拷贝就是对内存地址的复制,让目标对象指针和源对象指向同一片内存空间。

浅拷贝只是对对象的简单拷贝,让几个对象共用一片内存,当内存销毁的时候,指向这片内存的几个指针需要重新定义才可以使用,要不然会成为野指针。

iOS 里面的浅拷贝:

在 iOS 里面, 使用retain 关键字进行引用计数,就是一种更加保险的浅拷贝。他既让几个指针共用同一片内存空间,又可以在release 由于计数的存在,不会轻易的销毁内存,达到更加简单使用的目的。

二、深拷贝(克隆人)

紧接上面的例子: 创建一个dataArray的可变副本并将它赋值给array的最终副本,这就在内存中创建两个截然不同的可变数组.

array = [dataArray mutableCopy];
[array removeObjectAtIndex:0];


输出结果: dataArray : two,three,four

array: three,four

array则是我们所说的真正意义上的复制,系统为其分配了新内存,是两个独立的字符串内容是一样的。对array进行操作,不会影响dataArray。

深拷贝是指拷贝对象的具体内容,而内存地址是自主分配的,拷贝结束之后,两个对象虽然存的值是相同的,但是内存地址不一样,两个对象也互不影响,互不干涉。

iOS提供了copy和mutableCopy方法,顾名思义,copy就是复制了一个imutable的对象,而mutableCopy就是复制了一个mutable的对象。以下将举几个例子来说明。

这里指的是NSString, NSNumber等等一类的对象。

NSString *string = @”dddd";
NSString *stringCopy = [string copy];
NSMutableString *stringDCopy = [string mutableCopy];
[stringMCopy appendString:@"!!"];


查看内存可以发现,string和stringCopy指向的是同一块内存区域(weak reference),引用计数没有发生改变。而stringMCopy则是我们所说的真正意义上的复制,系统为其分配了新内存,是两个独立的字符串内容是一样的。

补充:

copy 与 retain 的区别:

copy 是创建一个新对象,retain 是创建一个指针,引用对象计数加一。 copy属性标识两个对象内容相同,新的对象retain count为1, 与旧有对象引用计数无关,旧有对象没有变化。copy减少对象对上下文的依赖。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: