重拾iOS-copy

image

关键词:深拷贝,浅拷贝,copy,mutableCopy,单层拷贝

面试题:

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

2)对可变对象进行copy是深拷贝还是浅拷贝?

3)为什么给NSString类型属性使用copy修饰,改为strong可以吗?

4)@property (copy) NSMutableArray *array; 这种写法有什么问题?

5)什么是单层拷贝,怎么实现多层拷贝?

6)修饰block属性时,为什么用copy?

一、深拷贝与浅拷贝

浅拷贝:

浅拷贝就是对内存地址的复制,让目标对象指针和源对象指向同一片内存空间,当内存销毁的时候,指向这片内存的几个指针需要重新定义才可以使用,要不然会成为野指针。

深拷贝:

深拷贝是指拷贝对象的具体内容,而内存地址是自主分配的,拷贝结束之后,两个对象虽然存的值是相同的,但是内存地址不一样,两个对象也互不影响,互不干涉。

【总结】

深拷贝就是内容拷贝,浅拷贝就是指针拷贝。本质区别在于:

  1. 是否开启新的内存地址;
  2. 是否影响内存地址的引用计数;

二、copy与mutableCopy

使用copy或mutableCopy方法可以创建一个对象的副本。

copy:

需要实现NSCoppying协议;

这些创建的是不可变副本(如NSString、NSArray、NSDictionary);

mutableCopy:

需要先实现NSMutableCopying协议;

创建的是可变副本(如NSMutableString、NSMutableArray、NSMutableDictionary);

三、单层拷贝与多层拷贝

1、普通对象的copy和mutableCopy

1)不可变对象copy
NSString *str = @"不可变对象";
NSString *copyStr = [str copy];
NSLog(@"str=%@, 地址:%p",str, str);
NSLog(@"copyStr=%@, 地址:%p",copyStr, copyStr);
// 打印:
2020-07-05 10:59:46.208267+0800 OCTestDemo[77626:1744180] str=不可变对象, 地址:0x108704330
2020-07-05 10:59:46.208372+0800 OCTestDemo[77626:1744180] copyStr=不可变对象, 地址:0x108704330
// 结论:对不可变对象copy,是浅拷贝。
2)不可变对象mutableCopy
NSString *str = @"不可变对象";
NSString *mtCopyStr = [str mutableCopy];
NSLog(@"str=%@, 地址:%p",str, str);
NSLog(@"mtCopyStr=%@, 地址:%p",mtCopyStr, mtCopyStr);
// 打印:
2020-07-05 11:01:20.982338+0800 OCTestDemo[77699:1746254] str=不可变对象, 地址:0x10fa86330
2020-07-05 11:01:20.982470+0800 OCTestDemo[77699:1746254] mtCopyStr=不可变对象, 地址:0x600000257fa0
// 结论:对不可变对象mutableCopy,是深拷贝。
3)可变对象copy
NSMutableString *str = [NSMutableString stringWithString:@"可变对象"];
NSString *copyStr = [str copy];
NSLog(@"str=%@, 地址:%p",str, str);
NSLog(@"copyStr=%@, 地址:%p",copyStr, copyStr);
// 打印:
2020-07-05 11:04:04.408830+0800 OCTestDemo[77768:1748628] str=可变对象, 地址:0x60000024c900
2020-07-05 11:04:04.408933+0800 OCTestDemo[77768:1748628] copyStr=可变对象, 地址:0x6000002279a0
// 结论:对可变对象copy,是深拷贝。
4)可变对象mutableCopy
NSMutableString *str = [NSMutableString stringWithString:@"可变对象"];
NSString *mtCopyStr = [str mutableCopy];
NSLog(@"str=%@, 地址:%p",str, str);
NSLog(@"mtCopyStr=%@, 地址:%p",mtCopyStr, mtCopyStr);
// 打印:
2020-07-05 11:05:38.838961+0800 OCTestDemo[77828:1750661] str=可变对象, 地址:0x600000256200
2020-07-05 11:05:38.839065+0800 OCTestDemo[77828:1750661] mtCopyStr=可变对象, 地址:0x60000025e8d0
// 结论:对可变对象mutableCopy,是深拷贝。

2、容器类对象的copy和mutableCopy

Animal *dog = [[Animal alloc]init];
Animal *cat = [[Animal alloc]init];
Animal *pig = [[Animal alloc]init];
1)不可变容器对象copy
NSArray *arr = @[dog, cat, pig];
NSArray *copyArr = [arr copy];
NSLog(@"arr=%@, 地址:%p",arr, arr);
NSLog(@"copyArr=%@, 地址:%p",copyArr, copyArr);
// 打印:
2020-07-05 11:10:12.324873+0800 OCTestDemo[77897:1754011] arr=(
    "<Animal: 0x600000015fc0>",
    "<Animal: 0x600000015670>",
    "<Animal: 0x600000015220>"
), 地址:0x6000004598f0
2020-07-05 11:10:12.325063+0800 OCTestDemo[77897:1754011] copyArr=(
    "<Animal: 0x600000015fc0>",
    "<Animal: 0x600000015670>",
    "<Animal: 0x600000015220>"
), 地址:0x6000004598f0
// 结论:对不可变容器对象copy,是浅拷贝,且为单层拷贝。
2)不可变容器对象mutableCopy
NSArray *arr = @[dog, cat, pig];
NSArray *mtCopyArr = [arr mutableCopy];
NSLog(@"arr=%@, 地址:%p",arr, arr);
NSLog(@"mtCopyArr=%@, 地址:%p",mtCopyArr, mtCopyArr);
// 打印:
2020-07-05 11:12:30.281808+0800 OCTestDemo[77952:1756412] arr=(
    "<Animal: 0x604000202810>",
    "<Animal: 0x6040000178c0>",
    "<Animal: 0x604000017c70>"
), 地址:0x60400024c7b0
2020-07-05 11:12:30.281930+0800 OCTestDemo[77952:1756412] mtCopyArr=(
    "<Animal: 0x604000202810>",
    "<Animal: 0x6040000178c0>",
    "<Animal: 0x604000017c70>"
), 地址:0x60400024d020
// 结论:对不可变容器对象mutableCopy,是深拷贝,且为单层拷贝。
3)可变容器对象copy
NSMutableArray *arr = [NSMutableArray arrayWithArray:@[dog, cat, pig]];
NSArray *copyArr = [arr copy];
NSLog(@"arr=%@, 地址:%p",arr, arr);
NSLog(@"copyArr=%@, 地址:%p",copyArr, copyArr);
// 打印:
2020-07-05 11:15:05.863855+0800 OCTestDemo[78006:1758411] arr=(
    "<Animal: 0x60400001e990>",
    "<Animal: 0x60400001e400>",
    "<Animal: 0x60400001e390>"
), 地址:0x60400024b040
2020-07-05 11:15:05.864007+0800 OCTestDemo[78006:1758411] copyArr=(
    "<Animal: 0x60400001e990>",
    "<Animal: 0x60400001e400>",
    "<Animal: 0x60400001e390>"
), 地址:0x60400024af50
// 结论:对可变容器对象copy,是深拷贝,且为单层拷贝。
4)可变容器对象mutableCopy
NSMutableArray *arr = [NSMutableArray arrayWithArray:@[dog, cat, pig]];
NSArray *mtCopyArr = [arr mutableCopy];
NSLog(@"arr=%@, 地址:%p",arr, arr);
NSLog(@"mtCopyArr=%@, 地址:%p",mtCopyArr, mtCopyArr);
// 打印:
2020-07-05 11:16:56.869198+0800 OCTestDemo[78056:1760281] arr=(
    "<Animal: 0x600000013270>",
    "<Animal: 0x600000012fb0>",
    "<Animal: 0x600000013b70>"
), 地址:0x6000004403c0
2020-07-05 11:16:56.869378+0800 OCTestDemo[78056:1760281] mtCopyArr=(
    "<Animal: 0x600000013270>",
    "<Animal: 0x600000012fb0>",
    "<Animal: 0x600000013b70>"
), 地址:0x600000440480
// 结论:对可变容器对象mutableCopy,是深拷贝,且为单层拷贝。

3、多层拷贝的实现方案

1)- (instancetype)initWithArray:(NSArray<ObjectType> *)array copyItems:(BOOL)flag;
NSMutableArray *arr = [NSMutableArray arrayWithArray:@[dog, cat, pig]];
NSArray *deepCopyArr = [[NSArray alloc]initWithArray:arr copyItems:YES];
NSLog(@"arr=%@, 地址:%p",arr, arr);
NSLog(@"deepCopyArr=%@, 地址:%p",deepCopyArr, deepCopyArr);
// 打印:
2020-07-05 11:27:19.171118+0800 OCTestDemo[78309:1769460] arr=(
    "<Animal: 0x600000000df0>",
    "<Animal: 0x6000000006b0>",
    "<Animal: 0x600000000cd0>"
), 地址:0x600000254850
2020-07-05 11:27:19.171313+0800 OCTestDemo[78309:1769460] deepCopyArr=(
    "<Animal: 0x600000000b70>",
    "<Animal: 0x600000000dc0>",
    "<Animal: 0x600000000d80>"
), 地址:0x6000002544f0
// 结论:使用initWithArray:copyItems:方法可以达到多层拷贝的效果,但这里也只是双重拷贝,并不是真正意义上的多层拷贝。
// 注意,数组内的元素需要实现NSCopying,NSMutableCopying协议。
2)归档解档大法
NSMutableArray *arr = [NSMutableArray arrayWithArray:@[dog, cat, pig]];
NSArray *deepCopyArr = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:arr]];
NSLog(@"arr=%@, 地址:%p",arr, arr);
NSLog(@"deepCopyArr=%@, 地址:%p",deepCopyArr, deepCopyArr);
// 打印:
2020-07-05 11:38:08.175207+0800 OCTestDemo[78604:1779461] arr=(
    "<Animal: 0x60000000ab90>",
    "<Animal: 0x60000000ab70>",
    "<Animal: 0x600000010bf0>"
), 地址:0x60000044f5a0
2020-07-05 11:38:08.175351+0800 OCTestDemo[78604:1779461] deepCopyArr=(
    "<Animal: 0x600000010f70>",
    "<Animal: 0x600000010ef0>",
    "<Animal: 0x600000010f60>"
), 地址:0x60000044f600
// 结论:使用归档解档方法可以达到多层拷贝的效果。注意,数组内的元素需要实现NSCoding协议。

总结

image

补充

1、为什么给NSString类型属性使用copy修饰,改为strong可以吗?

NSString、NSArray、NSDictionary等等经常使用copy关键字,是因为他们有对应的可变类型:NSMutableString、NSMutableArray、NSMutableDictionary;

copy 此特质所表达的所属关系与 strong 类似。然而设置方法并不保留新值,而是将其“拷贝” (copy)。 当属性类型为 NSString 时,经常用此特质来保护其封装性,因为传递给设置方法的新值有可能指向一个 NSMutableString 类的实例。

用 @property 声明 NSString、NSArray、NSDictionary 经常使用 copy 关键字,是因为他们有对应的可变类型:NSMutableString、NSMutableArray、NSMutableDictionary,他们之间可能进行赋值操作,为确保对象中的字符串值不会无意间变动,应该在设置新属性值时拷贝一份。

2、修饰block属性时,为什么用copy?

block 使用 copy 是从 MRC 遗留下来的“传统”,在 MRC 中,方法内部的 block 是在栈区的,使用 copy 可以把它放到堆区。在 ARC 中写不写都行:对于 block 使用 copy 还是 strong 效果是一样的,但写上 copy 也无伤大雅,还能时刻提醒我们:编译器自动对 block 进行了 copy 操作。

3、@property (copy) NSMutableArray *array; 这种写法有什么问题?

两个问题:

1、添加,删除,修改数组内的元素的时候,程序会因为找不到对应的方法而崩溃。因为 copy 就是复制一个不可变 NSArray 的对象;

2、使用了 atomic 属性会严重影响性能;

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,692评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,482评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,995评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,223评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,245评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,208评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,091评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,929评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,346评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,570评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,739评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,437评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,037评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,677评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,833评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,760评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,647评论 2 354