Objective-C——浅复制和深复制

关于copy

copy基本概念

什么是copy

copy的字面意思是“复制”、“拷贝”,是一个产生副本的过程

常见的复制有:文件复制

作用: 利用一个源文件产生一个副本文件

特点:

  1. 修改原文件的内容
  2. 修改副本文件的内容,不会影响源文件

OC中的copy

作用: 利用一个源对象产生一个副本对象

特点:

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

如何使用copy功能

  • 一个对象可以调用copymutableCopy方法来创建一个副本对象
  • copy:创建的是不可变副本(如NSStringNSArrayNSDictionary
  • mutableCopy:创建的是可变副本(NSMutableStringNSMutableDictionary

使用copy功能的前提

  • copy:需要遵守NSCopying协议,实现copyWithZone:方法
@protocol NSCopying
- (id)copyWithZone:(NSZone*)zone;
@end

使用mutableCopy功能的前提

  • copy:需要遵守NSMutableCopying协议,实现mutableCopyWithZone:方法
@protocol NSmutableCopying
- (id)mutableCopyWithZone:(NSZone*)zone;
@end

下面是调用mutableCopy的例子

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSString* str = @"Objective-C";
        NSMutableString* str2 = [str mutableCopy];
        NSLog(@"str = %@, str2 = %@", str, str2);
    }
    return 0;
}

上面代码中使用mutableCopy对str进行复制,所产生的新的对象副本类型是NSMutableString类型的,所以要新创建一个NSMutableString类型的str2,将刚才产生的副本存入

输出str和str2的值如下:

注意:

  1. 如果是通过不可变对象调用了copy方法,那么不会生成一个新的对象
    原因:因为原来的对象是不能修改的,拷贝出来的对象也是不能修改的,既然两个都不能修改,所以永远不能影响到另外一个对象,那么已经符合需求(OC为了对内存进行优化,就不会生成一个新的对象)
  2. OC中有部分对象已经遵守了NSCopyNSMutableCopy这两个协议,如NSString。但如果是自定义的对象,如自定义一个person*对象,并不遵守这两个协议。

深/浅复制

概念

正是因为调用copy方法有时候会生成一个新的对象,有时候不会生成一个新的对象

所以,
  如果没有生成新的对象,我们称之为浅复制,本质就是指针复制
  如果生成了新的对象,我们称之为深复制,本质就是会创建一个新的对象

浅复制

  • 浅复制(浅拷贝,指针拷贝,shallow copy)
  • 源对象和副本对象是同一个对象
  • 源对象(副本对象)引用计数器+1,相当于做一次retain操作
  • 本质是:没有产生新的对象

下列展示两个浅复制例子

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSString* str = @"Objective-C";
        NSString* str2 = [str copy];
        NSLog(@"str = %@, str2 = %@", str, str2);
    }
    return 0;
}

如上,保存的对象相同

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSArray* arr1 = [NSArray arrayWithObjects:@"苹果", @"桃子", @"葡萄", nil];
        NSArray* arr2 = [arr1 copy];
        NSLog(@"arr1 = %@, arr2 = %@", arr1, arr2);     //每个元素的指针地址相同
    }
    return 0;
}

如上,两个array对象地址相同

深复制

  • (深拷贝,内容拷贝,deep copy)
  • 源对象和副本对象是两个不同的对象
  • 源对象引用计数器不变,副本对象计数器为1(因为是新产生的)
  • 本质是:产生了新的对象

当创建一个NSArray数组

NSArray* arr1 = [NSArray arrayWithObjects:@"苹果", @"桃子", @"葡萄", nil];

mutbaleCopy复制一个不同类型的NSMutableArray数组

NSMutableArray* arr2 = [arr1 mutableCopy];

打印两者的指针地址,判断出是不相同的

//打印两者的指针地址
        NSLog(@"arr1 = %p, arr2 = %p", arr1, arr2);

在这里插入图片描述

打印arr1及arr2保存的元素,判断出是相同的

//打印arr1的元素
        NSLog(@"======打印arr1中的元素======");
        for (id object in arr1) {
            NSLog(@"%@", object);
        }
        //打印arr2的元素
        NSLog(@"======打印arr2中的元素======");
        for (id object in arr2) {
            NSLog(@"%@", object);
        }

上面就属于一个深复制:创建出了一个新的对象,指针地址自然就不同,但副本中的元素都和被复制的对象中的元素相同。

注意:

一般来说,深复制的实现难度大很多,尤其是当该对象包含大量的指针类型的实例变量时,如果某些实例变量里再次包含指针类型的实例变量,那么实现深复制会更加复杂

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值