博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Copy修饰的NSArray
阅读量:6991 次
发布时间:2019-06-27

本文共 3571 字,大约阅读时间需要 11 分钟。

深复制与浅复制

简单点理解,深复制,在内存中拷贝一份新的对象。 浅复制,没有拷贝新的对象,只是一个地址的引用。 在赋值过程中深复制操作,原对象的引用计数不会增加,浅复制引用计数会加一。

copy操作和mutableCopy操作

一个类遵循NSCopying,NSMutableCopying协议并且实现相对应的初始化方法后,这个类就具有对象拷贝的能力。如果你自定义类没有遵守协议直接调用copy/mutableCopy程序会奔溃。拷贝协议的具体使用我这里不做扩展感兴趣的可以自行Google一下。有一些刚接触iOS开发的同学认为copy操作就是浅复制,mutableCopy就是深复制,这是一个非常错误的理解。以下是我的一些总结:

  • 可变的集合对象 + copy 得到一个新的对象(新对象不可变)深复制
  • 可变的集合对象 + mutablecopy 得到一个新的对象(新对象可变)深复制
  • 不可变集合对象 + copy 没有得到新的对象(地址的引用)浅复制
  • 不可变集合对象 + mutablecopy 得到一个新的对象(新对象可变)深复制 OC中集合对象NSArray、NSMutableArray、NSDictionary、NSMutableDictionary、NSSet、NSMutableSet等,另外NSString、NSMutableString也准守以上原则。实践出真理我们来看看代码中的效果:
NSString *str = @"abc";    NSMutableString *mStr = [NSMutableString stringWithString:str];    NSString *strCopy = [str copy];    NSString *strMutableCopy = [str mutableCopy];    NSString *mStrCopy = [mStr copy];    NSString *mStrMutableCopy = [mStr mutableCopy];        NSArray *arr = @[@"abc"];    NSMutableArray *mArr = [NSMutableArray arrayWithArray:arr];    NSArray *arrCopy = [arr copy];    NSArray *arrMutableCopy = [arr mutableCopy];    NSArray *mArrCopy = [mArr copy];    NSArray *mArrMutableCopy = [mArr mutableCopy];        NSDictionary *dic = @{@"1":@"abc"};    NSMutableDictionary *mDic = [NSMutableDictionary dictionaryWithDictionary:dic];    NSDictionary *dicCopy = [dic copy];    NSDictionary *dicMutableCopy = [dic mutableCopy];    NSDictionary *mDicCopy = [mDic copy];    NSDictionary *mDicMutableCopy = [mDic mutableCopy];    NSSet *set = [NSSet setWithObject:@"1"];    NSMutableSet *mSet = [NSMutableSet setWithSet:set];    NSDictionary *setCopy = [set copy];    NSDictionary *setMutableCopy = [set mutableCopy];    NSDictionary *mSetCopy = [mDic copy];    NSDictionary *mSetMutableCopy = [mDic mutableCopy];复制代码

我们选取NSArray、NSMutableArray来看一下copy和mutableCopy的结果

stong与copy修饰符

strong和copy都可以用来修饰对象类型属性,strong修饰的属性,在将一个对象赋值给这个属性的时候,对象不会进行内存的拷贝,直接进行地址的引用,对象的引用计数+1;copy修饰的属性,如果该属性是一个非集合类型,那么赋值操作的效果和strong是一样的,如果该属性是一个集合类型,我们在下面在详细的解读。

NSMutableArray属性请使用strong修饰符

数学里面一个证明方式叫做反证法,接下来我们自定义一个类,增加一个NSMutableArray属性,使用copy来修饰

#import 
@interface TestObject : NSObject@property (nonatomic, copy) NSMutableArray *mutableArr;@end#import "TestObject.h"@implementation TestObject- (instancetype)init{ self = [super init];// self.mutableArr = [NSMutableArray arrayWithObject:@"1"]; _mutableArr = [NSMutableArray arrayWithObject:@"1"]; [self.mutableArr addObject:@"2"];//这一步,如果使用点语法进行属性赋值,程序crash;如果直接使用成员变量方式赋值,代码没有问题 return self;}//一般情况编译器会自动给属性添加get/set方法- (void)setMutableArr:(NSMutableArray *)mutableArr{ _mutableArr = [mutableArr copy];}@end复制代码

执行下面代码

TestObject *testObject = [TestObject new];testObject.mutableArr = [NSMutableArray arrayWithObject:@"1"];[testObject.mutableArr addObject:@"2"];复制代码

程序crash,控制台提示错误 [__NSSingleObjectArrayI addObject:]: unrecognized selector sent to instance 0x1c0017730 这一步我们就证明了如果我们使用copy来修饰一个可变的数组属性,然后通过点语法的方式把一个可变的数组赋值给该属性,最后该属性指向的是一个拷贝得到的全新的不可变数组,这个时候我们可以在代码中调用arrayWithObject等改变数组的api(因为@property (nonatomic, copy) NSMutableArray *mutableArr 告诉编译器这是一个可变的数组),但是在运行时,这个属性指向的是一个不可变的数组,引发unrecognized selector sent to instance 错误,从而导致crash。

NSArray属性请使用copy修饰符

我们定义一个NSArray类型的属性,目的是为了在初始化或者赋值操作后,这个数组集合就不能够改变的(增加或者删除元素等)。如果我们使用strong去修饰NSArray属性,并且我们把一个可变的数组NSMutableArray对象赋值给了该属性(父类的指针可以指向子类对象),strong并不会在内存中拷贝新的对象,只是一个地址的引用,所以NSArray类型的属性实质上指向的是一个NSMutableArray,这个时候如果其他地方还有一个NSMutableArray指针变量指向这个NSMutableArray对象,并且通过这个变量改变了数组,也会造成NSArray类型属性的改变,因为它们指向是一个地址内存,也就是一个对象。这就违背了我们设计的初衷,同时可能引发一些业务逻辑问题。

尾语

以上内容有任何的疑问、错误,请各路大神指出,同时希望能够帮助到大家

转载于:https://juejin.im/post/5a5dc908518825732335281c

你可能感兴趣的文章
安卓截屏如何实现将摄像头显示画面截下来
查看>>
jquery常识
查看>>
EF中的MySql返回 DataTable公共类库
查看>>
Visual Studio 2008常见问题
查看>>
【洛谷 P4254】 [JSOI2008]Blue Mary开公司(李超线段树)
查看>>
scrapy初体验 - 安装遇到的坑及第一个范例
查看>>
OC内存管理
查看>>
C#中Split用法
查看>>
3月6日 c#语言
查看>>
[LeetCode] Surrounded Regions, Solution
查看>>
MySQL系列:数据库基本操作(1)
查看>>
cpu真实核数
查看>>
hdu1058(dp)
查看>>
android EditText与TextView几个常用的属性
查看>>
SDN第五次上机作业
查看>>
redis 重要的配置参数
查看>>
Oracle 高级编程 01 ~
查看>>
JS重点整理之JS原型链彻底搞清楚
查看>>
springboot 配置文件
查看>>
浏览器插件 - Chrome 对 UserScript 的声明头(metadata)兼容性一览
查看>>