copy 和 mutableCopy 你真的理解吗?最近发现很多面试者基本都不能很好地回答这个问题。所以整理一下。

copy和mutableCopy的概念:

copy 浅拷贝,不拷贝对象本身,仅仅是拷贝指向对象的指针。

    NSString *str1 = @"str1";
NSString *str2 = [str1 copy];
NSLog(@"\nstr1 = %@ str1P = %p \n str2 = %@ str2P = %p", str1, str1, str2, str2); /*输出结果,可以得出下图结论
str1 = str1 str1P = 0x104d75180
str2 = str1 str2P = 0x104d75180
*/
 
copy网络图解.png

mutableCopy 深拷贝,是直接拷贝整个对象内存到另一块内存中。

    NSMutableString *mStr1 = [@"123" mutableCopy];
NSMutableString *mStr2 = [mStr1 mutableCopy];
NSLog(@"\n mStr1 = %@ mStr1P = %p \n mStr2 = %@ mStr2P = %p", mStr1, mStr1, mStr2, mStr2);
/*输出结果,可以得出下图结论
mStr1 = 123 mStr1P = 0x6000004460c0
mStr2 = 123 mStr2P = 0x600000446420
*/
 
mutableCopy网络图解.png

大部分人可能理解到这里就结束了。
当继续往下提问,如果是浅拷贝,改变str1的值,str2的值会变化吗?

问题1:上面浅拷贝的情况下,改变str1的值,str2的值会变化吗?

大部分人会不假思索的回答,因为str1 和 str2 指向同一个内存空间,str1变化,str2的值也会变化。

分析的很有道理,老铁看似没毛病毛病!

但是copy还有它的特点:

  • 修改源对象的属性和行为,不会影响副本对象
  • 修改副本对象的属性和行为,不会影响源对象

我们来试验一下:

    NSString *str1 = @"str1";
NSString *str2 = [str1 copy]; str1 = @"asdf"; NSLog(@"\nstr1 = %@ str1P = %p \n str2 = %@ str2P = %p", str1, str1, str2, str2); /*输出结果,修改str2 同理
str1 = asdf str1P = 0x10776b1a0
str2 = str1 str2P = 0x10776b180
*/

那为什么NSString *str2 = [str1 copy];是不同的指针指向同一块内存空间,str1 从新赋值 后两个内存空间就不一样了呢?

因为str2 = str1的时候,两个字符串都是不可变的,指向的同一块内存空间中的 @"str1",是不可能变成@"abcd"的。所以这个时候,为了优化性能,系统没必要另外提供内存,只生成另外一个指针,指向同一块内存空间就行。
但是当你从新给 str1 或者str2赋值的时候,因为之前的内容不可变,还有互不影响的原则下,这个时候,系统会从新开辟一块内存空间。

问题2:copy 一个可变的数组,会出现什么结果?

我们直接上结果吧。。。

    NSMutableArray *mArr1 = [@[@"123", @"456", @"asd"] mutableCopy];
NSMutableArray *mArr2 = [mArr1 copy]; NSLog(@"\n mArr1 = %@ mArr1P = %p mArr1 class = %@ \n\n mArr2 = %@ mArr2P = %p mArr2 class = %@", mArr1, mArr1, [mArr1 class], mArr2, mArr2, [mArr2 class]); /*输出结果
mArr1 = (
123,
456,
asd
)
mArr1P = 0x60400025db20
mArr1 class = __NSArrayM mArr2 = (
123,
456,
asd
)
mArr2P = 0x60400025dd30
mArr2 class = __NSArrayI
*/

从结果看出,内存地址不一样,而且mArr2 是不可变的。copy为什么不是指针指向了?
首先,mArr2是通过copy 得来的,关键点在于copy,和mArr1 无关,所以他是不可变的。
另外,mArr1指向的内存空间是可变的,如果对mArr1进行修改,同一内存空间的内容就会变化。 遵循相会不影响的原则,加上mArr2是不可变的,mArr1 的内存空间已经不合适,所以此时的 copy从新开辟内存空间。

问题3:用 copy 修饰 NSMutableArray @property (nonatomic, copy) NSMutableArray *mArr;,对mArr 赋值会有什么结果?

    NSArray *arr = @[@"123", @"456", @"asd"];
self.mArr = [arr mutableCopy];
NSLog(@"\n arrP = %p \n self.mArrP = %p, self.mArr class = %@", arr, self.mArr, [self.mArr class]); /*输出结果
arrP = 0x60000044d1a0
self.mArrP = 0x60000044d1a0, self.mArr class = __NSArrayI
*/

可以看出内存地址不一样,但是_mArr 是不可变的数组。
因为 _mArr声明的时候是用 copy修饰,那么就限制了他为不可变的数组。 赋值的时候是用mutableCopy,可变数组的复制方法,所以会从新分配内存。

    NSArray *arr = @[@"123", @"456", @"asd"];
self.mArr = arr;
NSLog(@"\n arrP = %p \n self.mArrP = %p, self.mArr class = %@", arr, self.mArr, [self.mArr class]); /*输出结果
arrP = 0x60400044ecd0
self.mArrP = 0x60400044ecd0, self.mArr class = __NSArrayI
*/

直接赋值,内存地址没变。

问题4:浅拷贝,不拷贝对象本身,仅仅是拷贝指向对象的指针。深拷贝,是直接拷贝整个对象内存到另一块内存中。 有什么看法?

浅拷贝,不拷贝对象本身,仅仅是拷贝指向对象的指针。不够严谨,在一些特殊请款下,还是会拷贝整个对象内存到另一块内存中。

总结:

  • 用copy修饰的 或者赋值的 变量肯定是不可变的。
  • 用copy赋值,要看源对象是否是可变的,来决定只拷贝指针,还是也拷贝对象到另一块内存空间
  • 对象之间mutableCopy赋值,肯定会拷贝整个对象内存到另一块内存中
  • 对象之间赋值之后,再改变,遵循互不影响的原则

iOS copy和mutableCopy 整理的更多相关文章

  1. iOS Copy 和 MutableCopy的区别 深浅拷贝的区别-供参考

    概述 对于系统的非容器类对象,对一不可变对象复制,copy是指针复制(浅拷贝)和mutableCopy就是对象复制(深拷贝).如果是对可变对象复制,都是深拷贝,但是copy返回的对象是不可变的. 对于 ...

  2. iOS copy 和 mutableCopy 学习

    (参考 iOS 52个技巧学习心得笔记 第二章 对象 , 消息, 运行期)的对象部分 关于Copy  有个经典问题”大部分的时候NSString的属性都是copy,那copy与strong的情况下到底 ...

  3. iOS copy&mutableCopy理解

    Copy&mutableCopy   通过copy方法可以创建可变或不可变对象的不可变副本,通过mutableCopy可以创建可变或不可变对象的可变副本. 拷贝分为浅拷贝和深拷贝: 浅拷贝:指 ...

  4. iOS之NSArray类簇简介-(copy、mutableCopy导致程序crash)

    1.前言 开发时常常用数组对数据进行处理,对NSMutableArray进行操作时经常导致程序崩溃,特研究一下NSArray的类簇!涉及__NSPlaceholderArray.__NSArray0. ...

  5. [Objective-C] Copy 和 MutableCopy

    看了几篇文章,因为文章很新手向,所以内容很繁琐.故整理一下重点,写了测试程序去了解几个知识点,不讨论基本概念.新博客wossoneri.com传送门 非集合类对象的copy与mutableCopy / ...

  6. iOS学习笔记-精华整理

    iOS学习笔记总结整理 一.内存管理情况 1- autorelease,当用户的代码在持续运行时,自动释放池是不会被销毁的,这段时间内用户可以安全地使用自动释放的对象.当用户的代码运行告一段 落,开始 ...

  7. 转载一篇关于ios copy的文章

    由于原文创作时间较早,一些内容不实用了,我对其进行了加工,去掉了一部分内容,添加了一点注释. 原文连接 http://www.cnblogs.com/ydhliphonedev/archive/201 ...

  8. iOS学习笔记总结整理

    来源:http://mobile.51cto.com/iphone-386851_all.htm 学习IOS开发这对于一个初学者来说,是一件非常挠头的事情.其实学习IOS开发无外乎平时的积累与总结.下 ...

  9. 【转】copy 和 mutablecopy (深拷贝和浅拷贝)

    阅读本文之前首先了解copy与retain的区别,如果有不正确的地方望大家多多指教: copy与retain的区别: copy是创建一个新对象,retain是创建一个指针,引用对象计数加1.Copy属 ...

随机推荐

  1. 004-行为型-01-策略模式(Strategy)

    一.概述 定义了一系列算法,并将每个算法封装起来,使他们可以相互替换,且算法的变化不会影响到使用算法的客户.需要设计一个接口,为一系列实现类提供统一的方法,多个实现类实现该接口,设计一个抽象类(可有可 ...

  2. boost写的异步客户端样例代码修改为支持断开重连的代码

    考虑到boost的工业级强度,因此就直接用了,代码的官方示例地址:https://www.boost.org/doc/libs/1_55_0/doc/html/boost_asio/examples/ ...

  3. 了解美杜莎(Medusa)

    (1).美杜莎介绍 Medusa(美杜莎)是一个速度快,支持大规模并行,模块化的暴力破解工具.可以同时对多个主机,用户或密码执行强力测试.Medusa和hydra一样,同样属于在线密码破解工具.Med ...

  4. Elasticsearch技术解析与实战 PDF (内含目录)

    Elasticsearch技术解析与实战                                  介绍: Elasticsearch是一个强[0大0]的搜索引擎,提供了近实时的索引.搜索.分 ...

  5. 使用idea创建简单的webservice服务

     New project: 生成HelloWorld.wsdl: 配置好tomcat后还需要加入 Axis 的库: 启动后,访问http://localhost:8080/services: 点击He ...

  6. Spring Cloud Eureka 服务发现 4.2

      在微服务架构中,服务发现可以说是最为核心和基础的模块,该模块主要用于实现各个微服务实例的自动化注册与发现.在Spring Cloud的子项目中,Spring Cloud Netflix提供了Eur ...

  7. [转载] HashMap的工作原理-hashcode和equals的区别

    目录 前言 为什么需要使用Hashcode,可以从Java集合的常用需求来描述: 更深入的介绍 先来些简单的问题 HashMap的0.75负载因子 总结 我在网上看到的这篇文章,介绍的很不错,但是我看 ...

  8. LODOP统计table自动分页后的每页的某列合计值

    LODOP中超文本会根据打印项高度或超过纸张,自动分页.(相关博文:Lodop打印控件 超文本自动分页.LODOP中ADD_PRINT_TABLE.HTM.HTML表格自动分页测试.Lodop打印表格 ...

  9. Flutter Bloc状态管理 简单上手

    我们都知道,Flutter中Widget的状态控制了UI的更新,比如最常见的StatefulWidget,通过调用setState({})方法来刷新控件.那么其他类型的控件,比如StatelessWi ...

  10. 关于Form、ModelForm的一些操作(持续更新)

    1.前端循环:后端传到前端的form是可以循环的,以此获得想要展示的元素 <form method="post" class="form-horizontal&qu ...