热门标签 | HotTags
当前位置:  开发笔记 > IOS > 正文

详解iOS的深浅拷贝

本文详细介绍了IOS中的三种拷贝方式,对iOS的深浅拷贝有疑问的朋友们可以参考下本文。

前言

OC对象的三种拷贝方式

OC的对象拷贝有如下三种方式,很多时候我们把深复制和完全复制混为一谈,其他他们是有区别的,具体如下

浅复制(shallow copy):在浅复制操作时,对于被复制对象的每一层都是指针复制。

深复制(one-level-deep copy):在深复制操作时,对于被复制对象,至少有一层是深复制。

完全复制(real-deep copy):在完全复制操作时,对于被复制对象的每一层都是对象复制。

两图以避之


理解深复制(mutableCopy)

浅复制很简单,就不演示了,看上面的图就懂了,只是简单的指针拷贝,所以改变原对象或者拷贝后的对象,都会影响另外一个对象。

从上图我们可以看到mutableCopy对于任何对象都是内容复制,也就是说进行了深复制。

上代码:

  NSMutableArray * dataArray1=[NSMutableArray arrayWithObjects:
                [NSMutableString stringWithString:@"1"],
                [NSMutableString stringWithString:@"2"],
                [NSMutableString stringWithString:@"3"],
                [NSMutableString stringWithString:@"4"],
                nil

                ];
  NSMutableArray * dataArray2=[NSMutableArray arrayWithObjects:
                [NSMutableString stringWithString:@"one"],
                [NSMutableString stringWithString:@"two"],
                [NSMutableString stringWithString:@"three"],
                [NSMutableString stringWithString:@"four"],
                dataArray1,
                nil
                ];

  NSMutableArray * dataArray3;
  NSMutableString * mStr;

  dataArray3=[dataArray2 mutableCopy];

  mStr = dataArray2[0];
  [mStr appendString:@"--ONE"];

  NSLog(@"dataArray3:%@",dataArray3);
  NSLog(@"dataArray2:%@",dataArray2);

输出如下:

2016-07-31 17:40:30.702 test1[2113:169774] dataArray3:(
  "one--ONE",
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)
2016-07-31 17:40:30.703 test1[2113:169774] dataArray2:(
  "one--ONE",
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)

看上面的输出,我们发现我们改变原数组dataArray2,竟然也会影响深复制后的dataArray3,不是说好的内容复制吗,为什么会这样?

这里我们来说说深复制和完全复制的区别

我们知道深复制,就是把原有对象的内容直接克隆一份到新对象,但是这里有一个坑就是他只会复制一层对象,而不会复制第二层甚至更深层次的对象。

代码dataArray3=[dataArray2 mutableCopy];只是对数组dataArray2本身进行了内容拷贝,但是里面的字符串对象却没有进行内容拷贝,而是进行的浅复制,那么dataArray2dataArray3里面的对象是共享同一份的。所以才会出现上面的情况。

单层深复制

那么如何解决上面的问题呢?

可以使用如下代码

  dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];

输出如下:

2016-07-31 17:45:48.472 test1[2151:173221] dataArray3:(
  one,
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)
2016-07-31 17:45:48.472 test1[2151:173221] dataArray2:(
  "one--ONE",
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)

可以看到dataArray3并没有被改变,但是别高兴的太早,我们再来改改。

代码如下:

  NSMutableArray * dataArray1=[NSMutableArray arrayWithObjects:
                [NSMutableString stringWithString:@"1"],
                [NSMutableString stringWithString:@"2"],
                [NSMutableString stringWithString:@"3"],
                [NSMutableString stringWithString:@"4"],
                nil

                ];
  NSMutableArray * dataArray2=[NSMutableArray arrayWithObjects:
                [NSMutableString stringWithString:@"one"],
                [NSMutableString stringWithString:@"two"],
                [NSMutableString stringWithString:@"three"],
                [NSMutableString stringWithString:@"four"],
                dataArray1,
                nil
                ];

  NSMutableArray * dataArray3;
  NSMutableString * mStr;

  dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];

  NSMutableArray *mArr = (NSMutableArray *)dataArray2[4];
  mStr = mArr[0];
  [mStr appendString:@"--ONE"];

  NSLog(@"dataArray3:%@",dataArray3);
  NSLog(@"dataArray2:%@",dataArray2);

输出如下:

2016-07-31 17:47:19.421 test1[2174:174714] dataArray3:(
  one,
  two,
  three,
  four,
    (
    "1--ONE",
    2,
    3,
    4
  )
)
2016-07-31 17:47:19.421 test1[2174:174714] dataArray2:(
  one,
  two,
  three,
  four,
    (
    "1--ONE",
    2,
    3,
    4
  )
)

可以看到深复制又失效了,这是因为dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];仅仅能进行一层深复制,对于第二层或者更多层的就无效了,那怎么办呢?

别急,我们还有大招没放。

完全复制

要想对多层集合对象进行复制,我们需要进行完全复制,这里可以使用归档和接档。

实现代码如下:

  dataArray3 = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:dataArray2]];

此时输出如下:

2016-07-31 17:49:55.561 test1[2202:177163] dataArray3:(
  one,
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)
2016-07-31 17:49:55.562 test1[2202:177163] dataArray2:(
  one,
  two,
  three,
  four,
    (
    "1--ONE",
    2,
    3,
    4
  )
)

可以看到dataArray3没有被dataArray2的修改影响。

类复制

说完了对象的复制,我们来看看如何实现类的复制,因为比较简单,直接放上代码

定义类复制

#import 
@interface Person : NSObject
@property(strong,nonatomic)NSString *age;
@property(strong,nonatomic)NSString *name;
@end
#import "Person.h"
@implementation Person
- (id)copyWithZone:(NSZone *)zone
{
  Person *person = [[Person allocWithZone:zone] init];
  person.age = self.age;
  person.name = self.name;
  return person;
}
@end

调用

  Person *person = [[Person alloc]init];
    person.age = @"dsdsd";
    person.name = @"dsdsdddww";

    Person *copyPerson = [person copy];
    NSLog(@"%@-----%@",copyPerson.age, copyPerson.name);

可以看到copyPerson的两个属性和persona一样。

@property中的copy关键字

在设置NSString类型的属性的时候,我们最好设置为copy类型,这样别人使用我们定义的属性的时候,他不管怎么改动该属性的赋值,都不会影响我们给该属性赋的值,为什么呢?

下面我们来看看


如上图所示,string2的属性是copy类型,可以看到是无法被修改的。

因为此时string2copystring的内存地址不一样,修改一个,不会影响另外一个。


上图所示,如果string2的属性是strong类型,就可以被修改,如下图所示:

因为此时string2copystring的内存地址都是一样的,修改一个,两个就同时被修改

copy关键字的NSMutableString崩溃

原因:

copy关键字的stringsetter方法实际上是把参数copy之后再赋值给变量_string,那么此时变量_string虽然被申明为NSMutableString,但是copy之后,就把 变量_string变成了不可变的NSString类型,所以就会出现方法报错,提示对不可变的NSString使用了NSMutableString的方法appendString

总结

以上就是iOS的深浅拷贝的详细内容,希望本文在大家开发iOS的过程中能有所帮助。


推荐阅读
  • EPICS Archiver Appliance存储waveform记录的尝试及资源需求分析
    本文介绍了EPICS Archiver Appliance存储waveform记录的尝试过程,并分析了其所需的资源容量。通过解决错误提示和调整内存大小,成功存储了波形数据。然后,讨论了储存环逐束团信号的意义,以及通过记录多圈的束团信号进行参数分析的可能性。波形数据的存储需求巨大,每天需要近250G,一年需要90T。然而,储存环逐束团信号具有重要意义,可以揭示出每个束团的纵向振荡频率和模式。 ... [详细]
  • IhaveconfiguredanactionforaremotenotificationwhenitarrivestomyiOsapp.Iwanttwodiff ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文介绍了如何找到并终止在8080端口上运行的进程的方法,通过使用终端命令lsof -i :8080可以获取在该端口上运行的所有进程的输出,并使用kill命令终止指定进程的运行。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 本文介绍了C++中省略号类型和参数个数不确定函数参数的使用方法,并提供了一个范例。通过宏定义的方式,可以方便地处理不定参数的情况。文章中给出了具体的代码实现,并对代码进行了解释和说明。这对于需要处理不定参数的情况的程序员来说,是一个很有用的参考资料。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • Mac OS 升级到11.2.2 Eclipse打不开了,报错Failed to create the Java Virtual Machine
    本文介绍了在Mac OS升级到11.2.2版本后,使用Eclipse打开时出现报错Failed to create the Java Virtual Machine的问题,并提供了解决方法。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • 本文介绍了MyBioSource转甲状腺素蛋白定量检测ELISA试剂盒的应用方法及特点。ELISA法作为一项新技术在免疫诊断中的应用范围不断扩大,不仅适用于多种病原微生物引起的传染病、非传染病的免疫诊断,也可用于大/小分子抗原的定量检测。ELISA法具有灵敏、特异、简单、快速、稳定及易于自动化操作等特点,是一种早期诊断的良好方法,也可用于血清流行病学调查。MyBioSource转甲状腺素蛋白定量检测ELISA试剂盒使用方法包括对血清和血浆的操作要求。 ... [详细]
  • Android Studio Bumblebee | 2021.1.1(大黄蜂版本使用介绍)
    本文介绍了Android Studio Bumblebee | 2021.1.1(大黄蜂版本)的使用方法和相关知识,包括Gradle的介绍、设备管理器的配置、无线调试、新版本问题等内容。同时还提供了更新版本的下载地址和启动页面截图。 ... [详细]
  • 本文介绍了一种划分和计数油田地块的方法。根据给定的条件,通过遍历和DFS算法,将符合条件的地块标记为不符合条件的地块,并进行计数。同时,还介绍了如何判断点是否在给定范围内的方法。 ... [详细]
author-avatar
爱情黄昏泪的诱惑_494
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有