iOS 深拷贝与浅拷贝

概念

浅拷贝

浅拷贝只是对对象指针进行拷贝,与原对象指针指向同一块内存,引用计数+1.

深拷贝

深拷贝会重新申请一块内存,对象指针指向新的内存地址。与原指针存储的数据内容相同,内存地址不一样。

iOS拷贝

iOS开发中,浅拷贝和深拷贝要更复杂一些,涉及到集合对象和非集合对象的copymutableCopy

  • 系统非集合对象:如NSStringNSIntegerNSNumber……
  • 自定义非集合对象:
  • 集合对象:如NSArrayNSDictionary……

系统非集合对象的copy与mutableCopy

系统非集合对象的copymutableCopy,遵循以下规则:

  • 可变对象的copymutableCopy方法都是深拷贝
  • 不可变对象的copy方法是浅拷贝,mutableCopy方法是深拷贝
  • copy方法返回的对象是不可变对象
示例
    //可变对象的拷贝,copy和mutableCopy都是深拷贝
    NSMutableString *str1 = [NSMutableString stringWithString:@"test"];
    NSMutableString *str2 = [str1 copy];
    //copy返回的是不可变对象,因此str2不能改变,会发生崩溃
    //[str2 appendString:@"test"];
    NSMutableString *str3 = [str1 mutableCopy];
    [str3 appendString:@"test"];
    NSLog(@"%@、 %@、 %@",str1,str2,str3);
    NSLog(@"%p、 %p、 %p",str1,str2,str3);
    
    
    NSString *str11 = @"test";
    //直接copy是浅拷贝
    NSMutableString *str12 = [str11 copy];
    //copy返回的是不可变对象,str12不能被修改,因此会发生崩溃
    //[str12 appendString:@"test"];
    //mutableCopy是深拷贝
    NSMutableString *str13 = [str11 mutableCopy];
    [str13 appendString:@"test"];
    NSLog(@"%@、 %@、 %@",str11,str12,str13);
    NSLog(@"%p、 %p、 %p",str11,str12,str13);

结果:

test、 test、 testtest
0x600001438570、 0x8abf33c3441514b6、 0x6000014385a0

test、 test、 testtest
0x107c64100、 0x107c64100、 0x6000014385d0

自定义对象的copy与mutableCopy

自定义对象需要遵循NSCopyingNSMutableCopying协议并实现- (id)copyWithZone:(NSZone *)zone- (id)mutableCopyWithZone:(NSZone *)zone方法,否则对对象进行copymutableCopy会报错。

示例
#import "Test.h"
#import <objc/runtime.h>

@interface Test ()<NSCopying,NSMutableCopying>

@end


@implementation Test

- (id)copyWithZone:(NSZone *)zone {
    Test *test = [[Test alloc]init];
    return test;
}
- (id)mutableCopyWithZone:(NSZone *)zone {
    Test *test = [[Test alloc]init];
    return test;
}
- (void)encodeWithCoder:(NSCoder *)aCoder {
    unsigned int count;
    Ivar *ivar = class_copyIvarList([self class], &count);
    for (int i = 0 ; i < count ; i++) {
        Ivar iv = ivar[i];
        const char *name = ivar_getName(iv);
        NSString *strName = [NSString stringWithUTF8String:name];
        //利用KVC取值
        id value = [self valueForKey:strName];
        [aCoder encodeObject:value forKey:strName];
    }
    free(ivar);
}
- (instancetype)initWithCoder:(NSCoder *)aDecoder {
    Test *test = [[Test alloc]init];
    if (self != nil) {
        unsigned int count = 0;
        Ivar *ivar = class_copyIvarList([self class], &count);
        for (int i= 0 ;i < count ; i++) {
            Ivar var = ivar[i];
            const char *keyName = ivar_getName(var);
            NSString *key = [NSString stringWithUTF8String:keyName];
            id value = [aDecoder decodeObjectForKey:key];
            [test setValue:value forKey:key];
        }
        free(ivar);
    }
    return test;
}

- (void)dealloc {
    NSLog(@"Test-dealloc");
}

集合对象的copy与mutableCopy

实际上,集合对象与非集合对象所遵循的规则基本上是一样的。

示例

    NSMutableArray *array1 = [NSMutableArray arrayWithObjects:@"a",@"b",@"c", nil];
    NSArray *array12 = [array1 copy];
    NSMutableArray *array13 = [array1 mutableCopy];
    NSLog(@"%p、 %p、 %p",array1,array12,array13);
    NSLog(@"%p 、%p、 %p",array1[0],array12[0],array13[0]);
        
    NSMutableArray *array2 = [NSMutableArray arrayWithObjects:[NSMutableString stringWithString:@"a"],@"b",@"c", nil];
    NSArray *array22 = [array2 copy];
    NSMutableArray *array23 = [array2 mutableCopy];
    NSLog(@"%p、 %p、 %p",array2,array22,array23);
    NSLog(@"%p、 %p、 %p",array2[0],array22[0],array23[0]);
    
    
    NSArray *array3 = @[@"a",@"b",@"c"];
    NSArray *array32 = [array3 copy];
    NSMutableArray *array33 = [array3 mutableCopy];
    NSLog(@"%p、 %p、 %p",array3,array32,array33);
    NSLog(@"%p、 %p、 %p",array3[0],array32[0],array33[0]);
        
    NSArray *array4 = @[[NSMutableString stringWithString:@"a"],@"b",@"c"];
    NSArray *array42 = [array4 copy];
    NSMutableArray *array43 = [array4 mutableCopy];
    NSLog(@"%p、 %p、 %p",array4,array42,array43);
    NSLog(@"%p、 %p、 %p",array4[0],array42[0],array43[0]);

结果:

0x6000001a42d0、 0x6000001a4300、 0x6000001a4330
0x10b1bd118、 0x10b1bd118、 0x10b1bd118

0x6000001a4390、 0x6000001a4480、 0x6000001a4600
0x6000001a4360、 0x6000001a4360、 0x6000001a4360

0x10b1bd5e0、 0x10b1bd5e0、 0x6000001a81e0
0x10b1bd118、 0x10b1bd118、 0x10b1bd118

0x6000001cf000、 0x6000001cf000、 0x6000001cfcc0
0x6000001cf3f0、 0x6000001cf3f0、 0x6000001cf3f0

说明可变对象的copymutableCopy方法都是深拷贝;不可变对象的copy方法是浅拷贝,mutableCopy方法是深拷贝。

集合对象的完全拷贝

通过对存储的第一个数据的地址对比发现,无论是深拷贝还是浅拷贝、存储的数据是可变还是不可变,拷贝后集合中存储的数据还是同一份数据。说明上面集合对象的深拷贝并不是严格意义上的深拷贝,而是单层深拷贝(对集合对象来说,深拷贝时只是将第一层对象进行了深拷贝,内部的对象仍然是浅拷贝)。集合对象的完全拷贝,就是集合中的每一层的元素都是深拷贝。

1.使用 initWith***: copyItems:YES 方法。

自定义对象需要实现copymutableCopy方法。

2.先将集合进行归档,然后再解档。

自定义对象需要实现归档和接档方法。

示例
    NSMutableArray *array1 = [NSMutableArray arrayWithObjects:@"a",@"b",@"c", nil];
    NSArray *array2 = [[NSArray alloc] initWithArray:array1 copyItems:YES];
    NSLog(@"%p 、%p",array1,array2);
    NSLog(@"%p、 %p",array1[0],array2[0]);
    
    NSMutableArray *array3 = [NSMutableArray arrayWithObjects:[NSMutableString stringWithString:@"a"],@"b",@"c", nil];
    NSArray *array4 = [[NSArray alloc] initWithArray:array3 copyItems:YES];
    NSLog(@"%p、 %p",array3,array4);
    NSLog(@"%p、 %p",array3[0],array4[0]);
    
    Test *test1 = [[Test alloc] init];
    NSMutableArray *array5 = [NSMutableArray arrayWithObjects:test1, nil];
    NSArray *array6 = [[NSArray alloc] initWithArray:array5 copyItems:YES];
    NSLog(@"%p、 %p",array5,array6);
    NSLog(@"%p、 %p",array5[0],array6[0]);

结果:

0x600002769830、 0x600002768390
0x1093400b8、 0x1093400b8
0x6000027699b0、 0x600002769950
0x600002769860、 0xddb0b57a0693f81b
0x600002769a70、 0x600002b34580
0x600002b34550、 0x600002b34570

————————————————
参考文档:
iOS中的深复制与浅复制

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

推荐阅读更多精彩内容

  • iOS中拷贝的目的在于和源数据互不影响,修改源数据不会影响拷贝出来的对象,同样修改拷贝出来的对象也不会影响到源数据...
    Good_Citizen阅读 506评论 0 4
  • 本文参考 iOS浅谈:深.浅拷贝与copy.strong这一篇总结的不太到位,可以继续读下一篇 在iOS中关于深拷...
    拧发条鸟xds阅读 523评论 0 3
  • 在工作中,有时会涉及到深拷贝和浅拷贝的内容,发现有些地方理解的不够透彻,所以在网上搜集资料总结一下,主要分四个方面...
    LeverTsui阅读 3,552评论 3 5
  • 浅拷贝:只创建一个新的指针,指向原指针指向的内存深拷贝:创建一个新的指针,并开辟新的内存空间,内容拷贝自原指针指向...
    小宝二代阅读 1,194评论 0 10
  • 一、概念与总结 1、浅拷贝 浅拷贝就是对内存地址的复制,让目标对象指针和源对象指向同一片内存空间,当内存销毁的...
    Andy_7020阅读 1,977评论 0 3