iOS基础之深复制与浅复制
来源:互联网 发布:编写c语言程序步骤 编辑:程序博客网 时间:2024/06/06 02:05
#import "Car.h"
int main(int argc, const char * argv[])
{
}
结果说明:将第一种又可分为两种,一种是不可变如NSArray,一种是可变如NSMutableArray。
a:对于不可变的对象,如果进行copy复制,那么此时的copy相当于retain,并没有开辟新的内存,但是如果是进行mutableCopy则是进行了新内存的开辟,此时只是计数加1而已;
b:对于可变对象,则不管是copy还是mutableCopy,都开辟了新的内存空间。
特别说明:尽管上面的b重新开辟了内存,但是其也只是浅复制,因为他只是复制了这个对象,而对于这个对象中的属性并没有进行复制。
对于第二种自定义的对象,要实现复制则要实现
#import
#import "Car.h"
int main(int argc, const char * argv[])
{
}
-----------------------------------------
#import
@interface Car : NSObject<</span>NSCopying,NSMutableCopying>
@property (nonatomic,retain) NSString *price;
@property (nonatomic,copy) NSString *name;
@end
---------------------------------------
#import "Car.h"
@implementation Car
@synthesize name = _name,price = _price;
-(id)copyWithZone:(NSZone *)zone{
}
-(id)mutableCopyWithZone:(NSZone *)zone{
}
@end
对于第一种,实现的是浅拷贝,因为在协议方法中,并没有进行属性的复制;
对于第二种,实现的其实也是浅拷贝,这其实就是上面的a中所说的情况,属性是NSString类型,进行copy其实就相当于retain
对于第三种,实现的则是深拷贝,这时候不管是对象还是对象的属性都开辟了新内存;
第一第二种打印结果:car地址:0x7fed71c143a0,carCopy地址:0x7fed71c14ac0,carPrice地址:0x10bf6a388,carCopyPrice地址:0x10bf6a388
第三种打印结果:2013-06-17 20:45:49.722 CopyTest[620:403] car地址:0x106f143a0,carCopy地址:0x106f14ad0,carPrice地址:0x106dd3388,carCopyPrice地址:0x106f14db0
如果不懂再来看一下下面的一段代码:
- #import <Foundation/Foundation.h>
- int main(int argc, const char * argv[])
- {
- @autoreleasepool {
- NSMutableArray *dataArray = [NSMutableArray arrayWithObjects:
- [NSMutableString stringWithString:@"one"],
- [NSMutableString stringWithString:@"two"],
- [NSMutableString stringWithString:@"three"],
- nil
- ];
- NSMutableArray *dataArray2;
- NSMutableString *mStr;
- NSLog(@"dataArray: ");
- for(NSString *elem in dataArray)
- NSLog(@" %@", elem);
- //执行一个拷贝,然后改变其中的一个字符串(浅复制)
- dataArray2 = [dataArray mutableCopy];
- //这种方式会同时改变连个数组中的对象
- mStr = [dataArray objectAtIndex:0];
- [mStr appendString:@"ONE"];
- NSLog(@"dataArray:");
- for(NSString *elem in dataArray)
- NSLog(@" %@",elem);
- NSLog(@"dataArray2:");
- for(NSString *elem in dataArray2)
- NSLog(@" %@",elem);
- [dataArray2 release];
- }
- return 0;
- }
使用上述方法的执行结果如下:
注意:原始数组及其副本中的第一个元素的值:他们都被修改了。为什么会产生这样的结果呢?
或许你能理解为什么dataArray的第一个元素发生改变,但是不明白为什么它的副本也会改变。
这与默认的浅复制方式有关。它意味着使用mutableCopy方式复制数组时,在内存中为新的数组对象分配了空间,并且将单个元素复制到新的数组中。然而将原始数组中每个元素复制到新位置意味着:仅将引用从一个数组元素复制到另一个数组元素。这样做的最终结果,就是这两个数组中的元素都指向内存中的同一个字符串。这与将一个对象赋值给另一个对象没有什么不同。
要为数组中每个元素创建完全不同的副本,需要执行所谓的深复制。这就意味着要创建数组中的每个对象内容的副本,而不仅是这些对象的引用的副本(并且考虑一下,如果一个数组中的元素本身是数组对象时,深复制意味着该如何处理?)
假设想要更改其中一个集合而不是他的副本,那么可能要为单个元素创建自己的副本。例如假设想要更改代码中的dataArray2的第一个元素,但不更改dataArray的第一个元素,可以创建一个新的字符串(使用stringWithString:之类的方法)并将它存储到dataArray2的第一个位置,如下所示:
- //这种方式只会改变一个数组中的对象,而对另外一个没有影响
- mStr = [NSMutableString stringWithString:[dataArray2 objectAtIndex:0]];
- [mStr appendString:@"ONE"];
- [dataArray2 replaceObjectAtIndex:0 withObject:mStr];
如果顺利的话,你会发现即使替换了数组中的对象之后,mStr和dataArray2的第一个元素仍指向内存中的同一个对象。这意味着随后在程序中对mStr做的任何修改,也将会更改数组 的第一个元素。如果这不是你想要的,则可以总是释放mStr,并分配新实例,因为对象会被replaceObject:atIndex:withObject:方法自动保持。
- iOS基础之深复制与浅复制
- java基础之深复制与浅复制
- iOS浅复制与深复制
- iOS中的浅复制与深复制
- iOS开发之深复制浅复制
- iOS深复制、浅复制与完全深度复制
- iOS深复制浅复制
- iOS 集合的深复制与浅复制
- iOS 集合的深复制与浅复制
- iOS 集合的深复制与浅复制
- iOS 集合的深复制与浅复制
- iOS 集合的深复制与浅复制
- iOS 集合的深复制与浅复制
- iOS 集合的深复制与浅复制
- iOS-深复制(mutableCopy)与浅复制(copy)
- iOS 集合的深复制与浅复制
- iOS 集合的深复制与浅复制
- iOS 集合的深复制与浅复制
- 增强MyEclipse的代码自动提示功能 .
- 编写 PHPUnit 测试
- hdu 2485 Destroying the bus stations (dfs+bfs)
- phpunit -assert系列函数
- oracle11g R2 RAC 卸载 grid
- iOS基础之深复制与浅复制
- 代码大全2_6——可以工作的类
- linux下使用静态库需要注意的几个问题
- 黑马程序员--面向对象之七:Collection集合框架
- C++中const型数据的小结
- 推荐python主要模块学习
- phpunit使用(项目中)
- XP搜索功能不能使用解决办法
- apache编译常见问题