iOS中的浅拷贝和深拷贝

拷贝协议

在iOS开发中,只有实现了NSCopying | NSMutableCopy协议的对象才支持copy | mutableCopy操作,发送copy消息实际上是调用协议中的 copyWithZone: 方法,而发送mutableCopy消息则调用的是 mutableCopyWithZone: 方法。

@protocol NSCopying

- (id)copyWithZone:(nullable NSZone *)zone;

@end

@protocol NSMutableCopying

- (id)mutableCopyWithZone:(nullable NSZone *)zone;

@end

浅拷贝和深拷贝本质

  • 浅拷贝:没有进行真正的复制,被复制的对象在内存中仍然只有一份,只是有两个对象指向该内存地址。
  • 深复制:被复制的对象有了一份拷贝,或者说克隆。两者的内存地址不同。


    浅拷贝-----------------------------------------深拷贝

iOS中的拷贝操作主要作用于两种类型:对象和容器。对象例如NSString,而容器则指集合类如NSArray,NSDictionary等。
因为容器中包含对象,所以拷贝容器的时候,也会涉及其中对象的拷贝,在iOS的默认实现中,无论容器类本身做的是深拷贝还是浅拷贝,容器内的对象都是浅拷贝,如果想实现容器内对象的深拷贝,需要自己实现拷贝方法。

代码验证

不可变对象的拷贝 NSString

NSString *str = @"不可变对象";
NSString *str1 = [str copy];
NSString *str2 = [str mutableCopy];
    
NSLog(@"原地址%p----原类型%@", str, [str class]);
NSLog(@"copy后的地址%p----copy后的类型%@", str1, [str1 class]);
NSLog(@"mutableCopy后的地址%p----mutableCopy后的类型%@", str2, [str2 class]);

//输出结果
原地址0x1033947f0----原类型__NSCFConstantString
copy后的地址0x1033947f0----copy后的类型__NSCFConstantString
mutableCopy后的地址0x600000250800----mutableCopy后的类型__NSCFString
  • 不可变对象copy是浅拷贝,mutableCopy是深拷贝
  • 不可变对象copy后的对象仍是不可变对象,mutableCopy后的对象是可变对象

可变对象的拷贝 NSMutableString

NSMutableString *str = [NSMutableString stringWithFormat:@"可变对象"];
NSMutableString *str1 = [str copy];
NSMutableString *str2 = [str mutableCopy];
    
NSLog(@"原地址%p----原类型%@", str, [str class]);
NSLog(@"copy后的地址%p----copy后的类型%@", str1, [str1 class]);
NSLog(@"mutableCopy后的地址%p----mutableCopy后的类型%@", str2, [str2 class]);

//输出结果
原地址0x60400005cfe0----原类型__NSCFString
copy后的地址0x604000221640----copy后的类型__NSCFString
mutableCopy后的地址0x60400005d2b0----mutableCopy后的类型__NSCFString
  • 可变对象copy和mutableCopy都是深拷贝
  • 可变对象copy和mutableCopy后的对象都是可变对象

不可变容器的拷贝 NSArray

NSMutableString *str1 = [NSMutableString stringWithFormat:@"可变对象"];
NSString *str2 = [NSString stringWithFormat:@"不可变对象"];
NSArray *array = [NSArray arrayWithObjects:str1, str2, nil];
NSArray *copyArray = [array copy];
NSArray *mutableCopyArray = [array mutableCopy];
    
NSLog(@"数组原地址%p----数组原类型%@", array, [array class]);
NSLog(@"copy后的地址%p----copy后的类型%@", copyArray, [copyArray class]);
NSLog(@"mutableCopy后的地址%p----mutableCopy后的类型%@", mutableCopyArray, [mutableCopyArray class]);
    
NSLog(@"数组中第一个元素原地址%p----数组中第一个元素原类型%@", array[0], [array[0] class]);
NSLog(@"数组中第二个元素原地址%p----数组中第二个元素原类型%@", array[1], [array[1] class]);
NSLog(@"数组中第一个元素copy后地址%p----数组中第一个元素copy后类型%@", copyArray[0], [copyArray[0] class]);
NSLog(@"数组中第一个元素copy后地址%p----数组中第一个元素copy后类型%@", copyArray[1], [copyArray[1] class]);
NSLog(@"数组中第一个元素mutableCopy后地址%p----数组中第一个元素mutableCopy后类型%@", mutableCopyArray[0], [mutableCopyArray[0] class]);
NSLog(@"数组中第一个元素mutableCopy后地址%p----数组中第一个元素mutableCopy后类型%@", mutableCopyArray[1], [mutableCopyArray[1] class]);

//输出结果
数组原地址0x604000032960----数组原类型__NSArrayI
copy后的地址0x604000032960----copy后的类型__NSArrayI
mutableCopy后的地址0x60400005c3e0----mutableCopy后的类型__NSArrayM

数组中第一个元素原地址0x60400005ba20----数组中第一个元素原类型__NSCFString
数组中第二个元素原地址0x60400005bfc0----数组中第二个元素原类型__NSCFString
数组中第一个元素copy后地址0x60400005ba20----数组中第一个元素copy后类型__NSCFString
数组中第一个元素copy后地址0x60400005bfc0----数组中第一个元素copy后类型__NSCFString
数组中第一个元素mutableCopy后地址0x60400005ba20----数组中第一个元素mutableCopy后类型__NSCFString
数组中第一个元素mutableCopy后地址0x60400005bfc0----数组中第一个元素mutableCopy后类型__NSCFString
  • 不可变容器内同时有可变对象和不可变对象,无论容器进行的是copy还是mutableCopy,其内部对象都是浅拷贝。
  • 而不可变容器本身,如果是copy,则是浅拷贝,如果是mutableCopy,则是深拷贝

可变容器的拷贝 NSMutableArray

NSMutableString *str1 = [NSMutableString stringWithFormat:@"可变对象"];
NSString *str2 = [NSString stringWithFormat:@"不可变对象"];
NSMutableArray *array = [NSMutableArray arrayWithObjects:str1, str2, nil];
NSMutableArray *copyArray = [array copy];
NSMutableArray *mutableCopyArray = [array mutableCopy];
    
NSLog(@"数组原地址%p----数组原类型%@", array, [array class]);
NSLog(@"copy后的地址%p----copy后的类型%@", copyArray, [copyArray class]);
NSLog(@"mutableCopy后的地址%p----mutableCopy后的类型%@", mutableCopyArray, [mutableCopyArray class]);
    
NSLog(@"数组中第一个元素原地址%p----数组中第一个元素原类型%@", array[0], [array[0] class]);
NSLog(@"数组中第二个元素原地址%p----数组中第二个元素原类型%@", array[1], [array[1] class]);
NSLog(@"数组中第一个元素copy后地址%p----数组中第一个元素copy后类型%@", copyArray[0], [copyArray[0] class]);
NSLog(@"数组中第一个元素copy后地址%p----数组中第一个元素copy后类型%@", copyArray[1], [copyArray[1] class]);
NSLog(@"数组中第一个元素mutableCopy后地址%p----数组中第一个元素mutableCopy后类型%@", mutableCopyArray[0], [mutableCopyArray[0] class]);
NSLog(@"数组中第一个元素mutableCopy后地址%p----数组中第一个元素mutableCopy后类型%@", mutableCopyArray[1], [mutableCopyArray[1] class]);

//输出结果
数组原地址0x60400005ce60----数组原类型__NSArrayM
copy后的地址0x60400002fdc0----copy后的类型__NSArrayI
mutableCopy后的地址0x60400005cd10----mutableCopy后的类型__NSArrayM
数组中第一个元素原地址0x60400005cb00----数组中第一个元素原类型__NSCFString
数组中第二个元素原地址0x60400005caa0----数组中第二个元素原类型__NSCFString
数组中第一个元素copy后地址0x60400005cb00----数组中第一个元素copy后类型__NSCFString
数组中第一个元素copy后地址0x60400005caa0----数组中第一个元素copy后类型__NSCFString
数组中第一个元素mutableCopy后地址0x60400005cb00----数组中第一个元素mutableCopy后类型__NSCFString
数组中第一个元素mutableCopy后地址0x60400005caa0----数组中第一个元素mutableCopy后类型__NSCFString
  • 可变容器无论是copy还是mutableCopy,都会返回一个新的容器
  • 但容器内部的对象,无论是可变对象还是不可变对象,都是浅拷贝

结论

深拷贝还是浅拷贝?

无论是对象还是容器:

  • 如果是不可变的,则copy是浅拷贝,mutableCopy是深拷贝
  • 如果是可变的,copy和mutableCopy都是深拷贝
  • 容器内部对象默认都是浅拷贝

返回的是可变还是不可变对象?

对于对象来说:

  • 不可变对象copy返回不可变,mutableCopy返回可变
  • 可变对象copy和mutableCopy都是返回可变对象

对于容器来说:

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

推荐阅读更多精彩内容