标签:

什么是深拷贝?什么是浅拷贝?

为什么经常看到字符串属性要这样定义,那个copy是神马意思?

@property(nonatomic,copy)NSString* name;

为什么下面的写法是错误的?

@property(nonatomic,copy)NSMutableString* name;

copyWithZone方法又到底是干嘛用的?

接下来,我们将一起,一步一步的去揭晓问题的答案。

Copy到底是个啥?

其实我们真的没必要把copy想的太高深。它之所以叫copy,其终极目的已不言而喻,不管我们承认与否,它就是和Ctrl+c和Ctrl+v的作用一样。拷贝一个副本出来,使得两个对象直接互不影响,完全独立。

深拷贝浅拷贝

copy分为两种,一种是可变copy:MutableCopy。使用它copy出来的对象是可以对其内容进行改变的。另外一种是不可变copy:copy。使用它复制出来的对象,其内容是不可以改变的。

MutableCopy:可变拷贝,其拷贝过程就是在内存中重新开辟一块区域,将对象复制一份放到这个区域。新对象是可以改变的,而且新对象的改变对源对象是没有影响的。

copy:不可变copy,同样也是要开辟一段内存空间给新对象,但新对象是不可以改变的。(特殊情况:不可变-》不可变的拷贝,是不创建新内存空间的)

(注:这里的可变copy和不可变copy是相对Foundation框架的类而言的,如果我们自己定义的类实现copy,一般都是可变的)。

以上说的这些和深拷贝浅拷贝有什么关系呢?我举一个例子大家立刻就明白了。

假设我们要对一个不可变的对象进行不可变copy(原来的对象不可变,新对象也不可变)。那么大家觉的,我们还有必要给新对象新建一块内存么?反正大家都不可以对这个对象进行改变,那就统一使用着一个不就可以了么?那么,ios系统怎么处理这个问题呢?引用计数器加1啊。是的,这就是浅拷贝。而需要给新对象开闭内存空间的,就是深拷贝。

好,接下来我们通过一段代码说明上述问题:

- (void)viewDidLoad {
[super viewDidLoad]; //创建一个可变的字符串
NSMutableString* name = [NSMutableString stringWithFormat:@"好梦园的兔子"];
NSLog(@"源字符串:%p-%@",name,name); //可变-》可变
NSMutableString* name1 = [name mutableCopy];
[name1 appendString:@"1"];
NSLog(@"可变字符串:%p-%@",name1,name1); //可变-》不可变
NSString* name2 = [name copy];
NSLog(@"不可变字符串:%p-%@",name2,name2); NSLog(@"___________不可变到不可变______________");
//不可变-》不可变
NSString* weibo = @"好梦园的兔子";
NSLog(@"源字符串:%p-%@",weibo,weibo);
NSString* weibo1 = [weibo copy];
NSLog(@"新字符串:%p-%@",weibo1,weibo1);
}

大家观察运行结果中的对象地址:

2015-08-15 15:25:12.045 copy的那些事[1207:108085] 源字符串:0x7fede25a2f50-好梦园的兔子
2015-08-15 15:25:12.046 copy的那些事[1207:108085] 可变字符串:0x7fede25a1690-好梦园的兔子1
2015-08-15 15:25:12.046 copy的那些事[1207:108085] 不可变字符串:0x7fede25a1b10-好梦园的兔子
2015-08-15 15:25:12.046 copy的那些事[1207:108085] ___________不可变到不可变______________
2015-08-15 15:25:12.047 copy的那些事[1207:108085] 源字符串:0x10abe2188-好梦园的兔子
2015-08-15 15:25:12.047 copy的那些事[1207:108085] 新字符串:0x10abe2188-好梦园的兔子

Copy属性

@property(nonatomic,copy)NSString* name;

我们大家在初学oc的时候,想必都听老师们说过这样一句话,创建属性的时候,遇到字符串括号里就写copy。但这究竟是为什么呢,我问老师,老师说他的老师就是这么说的。呵呵,开个玩笑。

我们先说明一点,这样做导致的结果就是,你给属性赋值的时候,会进行一次拷贝操作。至于目的嘛,接下来你会找到答案的。废话不多说,上代码:

/*
新建一个Student类,里面有两个属性,nameCopy和nameStrong。
我们接下来就要看看这里的copy到底有什么用
*/
@interface Student : NSObject
@property(nonatomic,copy)NSString* nameCopy;
@property(nonatomic,strong)NSString* nameStrong;
@end
- (void)viewDidLoad {
[super viewDidLoad];
//新建一个可变的字符串
NSMutableString* name = [NSMutableString stringWithFormat:@"好梦园的兔子"];
NSLog(@"源字符串:%p-%@",name,name); //创建Student对象
Student * student1 = [[Student alloc]init];
student1.nameCopy = name;//记住,此时会进行一次copy操作,因为(nonatomic,copy)
student1.nameStrong = name;
NSLog(@"nameCopy:%p-%@",student1.nameCopy,student1.nameCopy);
NSLog(@"nameStrong:%p-%@",student1.nameStrong,student1.nameStrong);
/*
通过上述代码,我们发现,(nonatomic,copy)修饰的属性,赋值时进行了copy操作,开辟了一块内存存放该对象属性
(nonatomic,strong)赋值时,对象属性和源字符串共用了一块内存,只是对源字符串引用计数器+1。
*/ NSLog(@"—————————————源字符串改变—————————————————————");
//对源字符串进行改变 看看nameCopy和nameStrong哪个会受到影响
[name setString:@"字符串改变了"];
NSLog(@"nameCopy:%p-%@",student1.nameCopy,student1.nameCopy);
NSLog(@"nameStrong:%p-%@",student1.nameStrong,student1.nameStrong);
}

运行结果:

2015-08-15 16:09:07.698 copy的那些事[1504:130239] 源字符串:0x7f830a42fb00-好梦园的兔子
2015-08-15 16:09:07.699 copy的那些事[1504:130239] nameCopy:0x7f830a42c340-好梦园的兔子
2015-08-15 16:09:07.699 copy的那些事[1504:130239] nameStrong:0x7f830a42fb00-好梦园的兔子
2015-08-15 16:09:07.699 copy的那些事[1504:130239] ———————————————————源字符串改变———————————————————
2015-08-15 16:09:07.699 copy的那些事[1504:130239] nameCopy:0x7f830a42c340-好梦园的兔子
2015-08-15 16:09:07.700 copy的那些事[1504:130239] nameStrong:0x7f830a42fb00-字符串改变了

问题:

下面的写法有问题么?为什么?(下篇博客会为大家做解答)

@property(nonatomic,copy)NSMutableString* name;

自定义类Copy

如果我们希望自己定义的类,也能使用copy方法,嗖嗖嗖就能复制一堆对象,那该如何操作呢。

假设我们直接使用copy

Student* student2 =[ student1 copy];

我们会发现,这时候程序会报错,说我们没有实现copyWithZone方法。这说明,我们是可以为自己的类定义copy方法的,只是要进行一些规范性的操作。

自定义类实现copy的步骤:

(1)遵守NSCoping协议
(2)实现copyWithZone方法。//参数zone基本不用,它的意思是指定该方法从始至终都在某一个区域分配内存
(所有copy最终都会调用这个方法)
{
  //方法内部进行以下操作
  //(1)实例化对象
  A* a = [A alloc ]init]; //一般正规写法是[[self.class alloc] init]  因为这样子类也可以复用该方法
  //(2)给属性赋值
  a.xx = xx;
  //(3)返回新对象
  return a;
}
上代码:
#import <Foundation/Foundation.h>

/*
创建一个Teacher类,并让该类实现copy方法
*/ //1.遵守NSCopying协议
@interface Teacher : NSObject<NSCopying>
@property(nonatomic,copy)NSString* name;
//2.实现copyWithZone方法;
-(id)copyWithZone:(NSZone *)zone;
@end
#import "Teacher.h"

@implementation Teacher
//实现copyWithZone方法
-(id)copyWithZone:(NSZone *)zone
{
Teacher* teacher = [[self.class alloc]init];
teacher.name = self.name;
return teacher;
}
@end
@implementation ViewController

- (void)viewDidLoad {
[super viewDidLoad];
Teacher* teacher = [[Teacher alloc]init];
teacher.name = @"好梦园的兔子"; Teacher* teacher1 = [teacher copy]; NSLog(@"teacher:%p-%@",teacher,teacher.name);
NSLog(@"teacher1:%p-%@",teacher1,teacher1.name);
}

运行结果:

2015-08-15 16:41:07.821 copy的那些事[1779:148657] teacher:0x7fd750d13ce0-好梦园的兔子
2015-08-15 16:41:07.822 copy的那些事[1779:148657] teacher1:0x7fd750d13cf0-好梦园的兔子

上述内容,是将我们自己定义的类实现了copy方法。那么这里有个问题,我可不可以让我自己定义的类实现mutableCopy呢?有没有NSMutableCopying协议呢?

还有,上述代码中,teacher.name 和teacher1.name 的地址其实是一样的。为什么?

最后一问答案比较简单,我就替大家回答了(copyWithZone方法中:tescher.name = self.name)

其他两问,按照实现copy的方法进行尝试,同样会很快得到答案的。

(完)

IOS中的深拷贝和浅拷贝的更多相关文章

  1. 浅谈Java中的深拷贝和浅拷贝(转载)

    浅谈Java中的深拷贝和浅拷贝(转载) 原文链接: http://blog.csdn.net/tounaobun/article/details/8491392 假如说你想复制一个简单变量.很简单: ...

  2. C语言中的深拷贝和浅拷贝

    //C语言中的深拷贝和浅拷贝 #define _CRT_SECURE_NO_WARNINGS #include<stdio.h> #include<stdlib.h> #inc ...

  3. 浅谈Java中的深拷贝和浅拷贝

    转载: 浅谈Java中的深拷贝和浅拷贝 假如说你想复制一个简单变量.很简单: int apples = 5; int pears = apples; 不仅仅是int类型,其它七种原始数据类型(bool ...

  4. 内功心法 -- Java中的深拷贝和浅拷贝

    写在前面的话:读书破万卷,编码如有神--------------------------------------------------------------------这篇博客主要来谈谈" ...

  5. **Python中的深拷贝和浅拷贝详解

    Python中的深拷贝和浅拷贝详解   这篇文章主要介绍了Python中的深拷贝和浅拷贝详解,本文讲解了变量-对象-引用.可变对象-不可变对象.拷贝等内容.   要说清楚Python中的深浅拷贝,需要 ...

  6. javascript中的深拷贝与浅拷贝

    javascript中的深拷贝与浅拷贝 基础概念 在了解深拷贝与浅拷贝的时候需要先了解一些基础知识 核心知识点之 堆与栈 栈(stack)为自动分配的内存空间,它由系统自动释放: 堆(heap)则是动 ...

  7. JavaScript中的深拷贝和浅拷贝!【有错误】还未修改!请逛其他园子!

    JavaScript中的深拷贝和浅拷贝! 浅拷贝 1.浅拷贝只是拷贝一层,更深层次对象级别的只拷贝引用.{也就是拷贝的是地址!简而言之就是在新的对象中修改深层次的值也会影响原来的对象!} // 2.深 ...

  8. 001 说说Python中的深拷贝和浅拷贝

    在Python编程中忽略深拷贝和浅拷贝可能会造成未知的风险. 比如我们打算保存一份原始对象的副本作为上一状态的记录,此后修改原始对象数据时,若是副本对象的数据也发生改变,那么这就是一个严重的错误. 注 ...

  9. IOS开发之深拷贝与浅拷贝(mutableCopy与Copy)详解

    copy与retain的区别: copy是创建一个新对象,retain是创建一个指针,引用对象计数加1.Copy属性表示两个对象内容相同,新的对象retain为1 ,与旧有对象的引用计数无关,旧有对象 ...

随机推荐

  1. MVC过滤器:过滤器执行顺序

    如果某个Action过滤器运用了多种过滤器,那么过滤器的执行顺序是如何呢? 规则一:不同类型的过滤器有一个先后顺序 即执行顺序是:授权过滤器->动作过滤器->结果过滤器->异常过滤器 ...

  2. .net core 中间件使用

    using System; using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; usi ...

  3. Flask 教程 第五章:用户登录

    本文翻译自The Flask Mega-Tutorial Part V: User Logins 这是Flask Mega-Tutorial系列的第五部分,我将告诉你如何创建一个用户登录子系统. 你在 ...

  4. 1-5-JS基础-数组应用及实例应用

    array 数组 一般简写arr 格式 var arr [ '第1个','第2个','第3个','第4个' ] 最后一个不要叫逗号 alert(arr.length) 弹出数组长度 4个 alert( ...

  5. Linux-正则、grep、sed学习笔记

    一.正则和grep 正则表达式的实现分成了两类: 基本正则表达式(BRE)和扩展的正则表达式(ERE). BRE 和 ERE 之间有什么区别呢?这是关于元字符的问题.BRE 可以辨别以下元字符: ^ ...

  6. 如何在mac版本的python里安装pip

    mac里面python自带easy_install,在终端里面执行sudo easy_install pip.运行完可以用pip help测试一下是否安装成功,成功安装后,直接pip install ...

  7. MySQL的高级应用之Explain(完美详细版,看这一篇就够了)

    原文链接: https://blog.csdn.net/wx1528159409/article/details/83819985

  8. Linux:源代码安装及脚本安装的使用

    由于这两个安装方法比较少,就没有单独分开来写 源代码安装 源码安装的步骤 (1)对下载的码包进行解压 (2)进入解压目录执行 configure 命令做相关设置 (3)执行 make 进行编译 (4) ...

  9. 搭建 Optix 环境

    我参考了 第0个示例 OptixHello 学习Optix的工程配置以及基本框架 的配置过程,该文对于 Optix 的框架介绍的很好,但是按照该文配置遇到了一些问题,我花费了一番功夫自己摸索终于配置好 ...

  10. NOIP 2012 文化之旅

    洛谷 P1078 文化之旅 洛谷传送门 JDOJ 1788: [NOIP2012]文化之旅 T4 JDOJ传送门 Description Input Output Sample Input Input ...