OC语言-08-深拷贝与浅拷贝详解(示例)
概述
- 拷贝:复制一个与源对象内容相同的对象
- 实现拷贝,需要遵守以下两个协议
- NSCopying
- NSMutableCopying
- 拷贝返回对象的种类
- 可变,mutableCopy消息返回的对象
- 不可变,copy消息返回的对象
- 拷贝的种类
- 浅拷贝,只是复制了一个指向源对象的指针,未创建对象,未分配内存
- 深拷贝,复制了源对象,创建了新对象,分配了内存
- 注意
- 系统对容器类的对象与非容器类的对象的内存处理是不同的,即当一个没有被其他对象强引用的对象从容器中移除后,该对象就销毁
Copy与Retain
- copy
- 是创建一个新的对象,内容拷贝
- copy表示的是两个对象的内容相同, 新对象的引用计数为1
- 与旧对象的引用计数无关,就对象没有变化
- copy减少了对象对上下文的
- retain
- 创建的是一个指针,指针拷贝
- 对象地址相同,内容固然相同
- 对象的引用计数+1
不同对象的拷贝行为
非容器对象(
如NSString))- 对于不可变对象
规则
- copy,浅拷贝(指针复制)
- mutableCopy,深拷贝(对象复制),返回对象可变(产生新的 可变对象)
示例
- (void)imutableInstanceCopy
{
NSString *string = @"Welcome to Xcode"; //copy,浅拷贝
NSString *stringCopy = [string copy];
//mutableCopy,返回的对象可变
NSMutableString *stringMutableCopy = [string mutableCopy];
[stringMutableCopy appendString:@"!"]; //string与stringCopy的内存地址相同
NSLog(@"string: %p", string);
NSLog(@"strongCopy: %p", stringCopy); //string与stringMutableCopy的内存地址不同,分配了新的内存
NSLog(@"stringMCopy:%p", stringMutableCopy);
}
- 对于可变对象
规则
- copy,深拷贝(对象复制),返回对象不可变
- mutableCopy,深拷贝(对象复制)
示例
- (void)mutableInstanceCopy
{
NSMutableString *mutableString = [NSMutableString stringWithString: @"Welcome to Xcode"]; //深拷贝,返回对象不可变
NSString *stringCopy = [mutableString copy];
NSMutableString *mutableStringCopy = [mutableString copy];
//运行时,此句会报错,错误信息“Attempt to mutate immutable object with appendString:”
[mutableStringCopy appendString:@"~~~"]; //深拷贝,返回对象可变
NSMutableString *stringMutableCopy = [mutableString mutableCopy];
[stringMutableCopy appendString:@"!"]; //三者与mutableString的内存地址都不同
NSLog(@"mutableString: %p", mutableString);
NSLog(@"string: %p", stringCopy);
NSLog(@"mutableStringCopy: %p", mutableStringCopy);
NSLog(@"stringMutbleCopy:%p", stringMutableCopy);
}
- 对于不可变对象
容器对象(
NSArray)遵循非容器对象的拷贝原则
注意
容器内的元素是指针赋值(浅拷贝)
示例
- (void)containerInstanceShallowCopy
{
NSArray *array = [NSArray arrayWithObjects:[NSMutableString stringWithString:@"Welcome"],@"to",@"Xcode",nil]; //浅拷贝
NSArray *arrayCopy = [array copy];
//深拷贝
NSMutableArray *arrayMutableCopy = [array mutableCopy]; NSLog(@"array: %p", array);
NSLog(@"arrayCopy: %p", arrayCopy);
NSLog(@"arrayMutableCopy: %p", arrayMutableCopy); //容器内的对象是浅拷贝,即它们在内存中只有一份
NSMutableString *testString = [array objectAtIndex:0];
[testString appendString:@" you"]; //三个数组的内容同时改变
NSLog(@"array[0]: %@", array[0]);
NSLog(@"arrayCopy[0]: %@", arrayCopy[0]);
NSLog(@"arrayMutableCopy[0]: %@", arrayMutableCopy[0]);
}
实现真正意义上的深复制
- (void)containerInstanceDeepCopy
{
NSArray *array = [NSArray arrayWithObjects:[NSMutableString stringWithString:@"Welcome"],@"to",@"Xcode",nil]; //数组内对象是指针复制
NSArray *deepCopyArray = [[NSArray alloc] initWithArray:array];
//真正以上的深复制,数组内对象是对象复制
NSArray *trueDeepCopyArray = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:array]]; NSLog(@"array: %p", array);
NSLog(@"deepCopyArray: %p", deepCopyArray);
NSLog(@"trueDeepCopyArray: %p", trueDeepCopyArray); //改变array的第一个元素
[[array objectAtIndex:0] appendString:@" you"]; //只影响deepCopyArray数组的第一个元素
NSLog(@"array[0]: %@", array[0]);
NSLog(@"arrayCopy[0]: %@", deepCopyArray[0]);
//不影响trueDeepCopyArray数组的第一个元素,是真正意义上的深拷贝
NSLog(@"arrayMutableCopy[0]: %@", trueDeepCopyArray[0]);
}
自定义对象
- 在定义对象要实现拷贝,需要遵守NSCoping与NSMutableCoping协议,并实现以下方法
- - (id)copyWithZone:(NSZone *)zone,可变拷贝
- - (id)mutableCopyWithZone:(NSZone *)zone,不可变拷贝
- 示例(自定对象Person的拷贝)
遵守协议,设置成员属性
@interface Person : NSObject <NSCopying, NSMutableCopying>
/**姓名*/
@property (nonatomic, copy) NSMutableString *name;
/**地址*/
@property (nonatomic, copy) NSString *address;
/**年龄*/
@property (nonatomic, assign) NSInteger age;
@end
重写初始化方法
- (instancetype)init
{
if (self = [super init])
{
self.name = [[NSMutableString alloc] initWithString:@"XiaoYaowang"];
self.address = @"世俗孤岛";
self.age = 3;
}
return self;
}
实现- (id)copyWithZone:(NSZone *)zone
- (id)copyWithZone:(NSZone *)zone
{
Person *p = [[[self class] allocWithZone:zone] init];
p.name = [self.name copy];
p.address = [self.address copy];
p.age = self.age; return p;
}
实现- (id)mutableCopyWithZone:(NSZone *)zone
- (id)mutableCopyWithZone:(NSZone *)zone
{
Person *p = [[[self class] allocWithZone:zone] init];
//注意,此处是mutableCopy方法
p.name = [self.name mutableCopy];
p.address = [self.address copy];
p.age = self.age; return p;
}
- 在定义对象要实现拷贝,需要遵守NSCoping与NSMutableCoping协议,并实现以下方法
OC语言-08-深拷贝与浅拷贝详解(示例)的更多相关文章
- OC细节 - 1.深拷贝与浅拷贝详解
概述 拷贝:复制一个与源对象内容相同的对象 实现拷贝,需要遵守以下两个协议 NSCopying NSMutableCopying 拷贝返回对象的种类 可变,mutableCopy消息返回的对象 不可变 ...
- 【转】 c++拷贝构造函数(深拷贝,浅拷贝)详解
c++拷贝构造函数(深拷贝,浅拷贝)详解 2013-11-05 20:30:29 分类: C/C++ 原文地址:http://blog.chinaunix.net/uid-28977986-id-3 ...
- 【C++】拷贝构造函数(深拷贝,浅拷贝)详解
一.什么是拷贝构造函数 首先对于普通类型的对象来说,它们之间的复制是很简单的,例如: ; int b = a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量. 下面看一个类对 ...
- **Python中的深拷贝和浅拷贝详解
Python中的深拷贝和浅拷贝详解 这篇文章主要介绍了Python中的深拷贝和浅拷贝详解,本文讲解了变量-对象-引用.可变对象-不可变对象.拷贝等内容. 要说清楚Python中的深浅拷贝,需要 ...
- javascript实现引用数据类型的深拷贝和浅拷贝详解
关于引用类型值的详解,请看另一篇随笔 https://www.cnblogs.com/jinbang/p/10346584.html 深拷贝和浅拷贝,也就是引用数据类型栈和堆的知识点.深浅拷贝的原型都 ...
- c++拷贝构造函数(深拷贝,浅拷贝)详解
一.什么是拷贝构造函数 首先对于普通类型的对象来说,它们之间的复制是很简单的,例如: ; int b=a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量. 下面 ...
- python的拷贝方式以及深拷贝,浅拷贝详解
python的拷贝方法有:切片方法, 工厂方法, 深拷贝方法, 浅拷贝方法等. 几种方法都可以实现拷贝操作, 具体区别在于两点:1.代码写法不同. 2.内存地址引用不同 代码演示: import co ...
- C++ 深拷贝和浅拷贝详解
前言 在对象拷贝过程中,如果没有自定义拷贝构造函数,系统会提供一个缺省的拷贝构造函数,缺省的拷贝构造函数对于基本类型的成员变量,按字节复制,对于类类型成员变量,调用其相应类型的拷贝构造函数. 位拷贝( ...
- Go语言Slice作为函数参数详解
Go语言Slice作为函数参数详解 前言 首先要明确Go语言中实质只有值传递,引用传递和指针传递是相对于参数类型来说. 个人认为上诉的结论不对,把引用类型看做对指针的封装,一般封装为结构体,结构体是值 ...
随机推荐
- PEP8 Python 编码规范
一 代码编排1 缩进.4个空格的缩进(编辑器都可以完成此功能),不使用Tap,更不能混合使用Tap和空格.2 每行最大长度79,换行可以使用反斜杠,最好使用圆括号.换行点要在操作符的后边敲回车.3 类 ...
- 第九篇 SQL Server代理了解作业和安全
本篇文章是SQL Server代理系列的第九篇,详细内容请参考原文 在这一系列的上一篇,学习了如何在SQL Server代理作业步骤启动外部程序.你可以使用过时的ActiveX系统,运行批处理命令脚本 ...
- Google FlatBuffers——开源、跨平台的新一代序列化工具
前段时间刚试用了一个序列化工具cereal,请看cereal:C++实现的开源序列化库,打算再总结下我对google proto buf序列化库的使用呢, 结果还没动手,大Google又出了一个新的. ...
- SpringMVC基础——@ModelAttribute和@SessionAttribute
一.@ModelAttribute 注解 对方法标注 @ModelAttribute 注解,在调用各个目标方法前都会去调用 @ModelAttribute 标记的注解.本质上来说,允许我们在调用目标方 ...
- 关于 getWriter() has already been called for this response 的错误解决办法
java.lang.IllegalStateException: getWriter() has already been called for this response 今天在做显示图片的时候,( ...
- SQL Server存储过程复习(一)
--存储过程学习篇 --.简单存储过程不带参数的学习 IF OBJECT_ID('Orders_GetAllOrders','P') IS NOT NULL DROP PROCEDURE Orders ...
- C#设计模式——抽象工厂模式(Abstract Factory Pattern)
一.概述在软件开发中,常常会需要创建一系列相互依赖的对象,同时,由于需求的变化,往往存在较多系列对象的创建工作.如果采用常规的创建方法(new),会造成客户程序和对象创建工作的紧耦合.对此,抽象工厂模 ...
- 【原创】网站抓包HttpWebRequest不返回Javascript生成的Cookie的解决办法
前言: 最近在做中国移动爬虫的过程中,首先遇到的就是 在某个请求中,有一个名为“WT_PFC"的cookie键值是由前端JavaScript生成的,没有进入到HttpWebResponse中 ...
- 基于吉日嘎底层架构的Web端权限管理操作演示-用户管理
相信博客园的朋友对吉日嘎拉都不陌生,相信很多人也买了他的源码,应用于自己的项目. 但是你有没有过一个困惑? 那就是:没有一个基于网页的权限管理界面. 今天,这一切都不再是问题,我花了3年时间研究学习并 ...
- 多余的Using Namespaces或引用会影响程序的执行效率么?
在.NET程序编写中,需要using相应命名空间或添加相应的References,可有时候没有使用到的命名空间也被添加到了Using Namespaces中,那么,这样会影响程序的执行效率么? 通过示 ...