Objective-C 深拷贝与浅拷贝

深拷贝与浅拷贝

浅拷贝
  • 浅拷贝 (shallow copy)     特点:不会复制所引用的对象,其新复制的对象只是一个指针,指向现有的引用对象上.(其实就相当于 Windows 系统下的快捷方式的复制)

    下面举一个简单地例子来分别下深拷贝与浅拷贝:

//创建两个可变的字符串
NSMutableString *student = [[NSMutableString alloc] initWithString:@"ZhangSan"];
        NSMutableString *anotherStu= [[NSMutableString alloc] initWithString:@"LiSi"];
//     使用两个可变的字符串来创建数组,也就是组里的有两个元素,其元素类型是可变字符串
        NSArray *students = @[student,anotherStu];
//     用创建好的数组copy一份赋值给另外一个数组
        NSArray *copyStu = [students copy];
//     打印原数组和复制得到的数组的内容
        NSLog(@"before moidfy the students:%@ and copyStu:%@",students,copyStu);
//     修改复制后得到的数组的第一个元素的值
        [copyStu[0] appendString:@"-TeacherZhang"];
//     打印原数组的值, 查看是否发生了变化
        NSLog(@"after modify copyStu:  the students info:%@",students);

打印出结果是before moidfy the students:(

ZhangSan,

LiSi

) and copyStu:(

ZhangSan,

LiSi

)

after modify copyStu: the students info:(

“ZhangSan-TeacherZhang”,

LiSi

)

这就是浅拷贝,只是拷贝了一个指针,并没有拷贝一个完整的对象.

深拷贝
  • 深拷贝 (deep copy)

特点:真正意义上的复制概念.得到的结果是多个,而非只是对象的引用. (对象的完全复制,包括其中的方法)

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[]) {
    @autoreleasepool {
//      定义了一个不可变字符串
        NSString *str = @"hello";
        //打印出 str 的地址
        NSLog(@"str‘s address:%p",str);
        //通过mutableCopy协议可以将不可变的字符串转化为可变字符串
        NSMutableString *anotherStr = [str mutableCopy];
        NSLog(@"anotherStr‘s address:%p",anotherStr);

        [anotherStr appendString:@"qingyun"];
        NSLog(@"anotherString:%@",anotherStr);

        NSMutableString *mStr = [[NSMutableString alloc] initWithString:@"HELLO"];
        NSLog(@"mStr‘s address:%p",mStr);
//        通过copy协议可以将可变字符串转换为不可变字符串
        NSMutableString *anotherMStr = [mStr copy];
        NSLog(@"anotherMStr‘s address:%p",anotherMStr);
        [anotherMStr appendString:@"-QINGYUN"];

    }
    return 0;
}
自定义对象的拷贝

实现自定义对象的拷贝的前提是必须要实现协议,否则,自定义对象调用 copy 对象的时候会发生崩溃的现象.

实例来说明这个问题,创建一个 QYCar 的类继承自 NSObject, 且遵从 NSCopying 协议,在 main.m 中实现自定义对象的 copy

在 QYCar 的头文件中,代码如下:

#import <Foundation/Foundation.h>

@interface QYCar : NSObject <NSCopying>

@property (nonatomic, strong)NSString *name;
@property (nonatomic, assign)NSInteger year;

@end

在 QYCar.m 中的代码如下:

#import "QYCar.h"

@implementation QYCar

//开箱即用的初始化方式
- (instancetype)init
{
    self = [super init];
    if (self) {
        self.name = @"BWM";
        self.year = 2013;
    }
    return self;
}

- (NSString *)description
{
    return [NSString stringWithFormat:@"name:%@,year:%ld", self.name,self.year];
}

- (id)copyWithZone:(NSZone *)zone
{
    QYCar *carCopy;

    carCopy = [QYCar allocWithZone:nil];
    carCopy.name = self.name;
    carCopy.year = self.year;

    return carCopy;
}

@end

在 main.m 中建立一个对象 car, 并调用开箱即用的初始化方式,然后再新建一个对象 carCopy, 并把 car 的内容 copy 给 carCopy, 为了显示我们成功拷贝了 car, 更改 carCopy 的 name 属性,并打印出来,显示下我们这是深拷贝.

int main(int argc, const char * argv[]) {
    @autoreleasepool {

        QYCar *car = [[QYCar alloc] init];
        NSLog(@"car:%@",car);

        QYCar *carCopy = [car copy];
        carCopy.name = @"HeNanCar";
        NSLog(@"carCopy:%@",carCopy);
        NSLog(@"car again : %@",car);
     }
     return 0;
}     

输出的结果是:

2015-07-01 21:06:43.302 SelfWithCopy[2631:2055248] car:name:BWM,year:2013

2015-07-01 21:06:43.303 SelfWithCopy[2631:2055248] carCopy:name:HeNanCar,year:2013

2015-07-01 21:06:43.303 SelfWithCopy[2631:2055248] car again : name:BWM,year:2013

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-13 16:28:51

Objective-C 深拷贝与浅拷贝的相关文章

【转载】C++拷贝构造函数(深拷贝,浅拷贝)

对于普通类型的对象来说,它们之间的复制是很简单的,例如:int a=88;int b=a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量.下面看一个类对象拷贝的简单例子. #include <iostream>using namespace std;class CExample {private:     int a;public:     CExample(int b)     { a=b;}     void Show ()     {        cout<

完全理解python深拷贝和浅拷贝

import copya = [1, 2, 3, 4, ['a', 'b']]  #原始对象b = a  #赋值,传对象的引用c = copy.copy(a)  #对象拷贝,浅拷贝d = copy.deepcopy(a)  #对象拷贝,深拷贝a.append(5)  #修改对象aa[4].append('c')  #修改对象a中的['a', 'b']数组对象print 'a = ', aprint 'b = ', bprint 'c = ', cprint 'd = ', d 输出结果:a = 

javaScript之深拷贝与浅拷贝

js中有两种数据类型: 1.  基本类型  :  Number.String.Boolean.Null.Undefined 2.  复杂类型  :  Object .Array 深拷贝和浅拷贝只针对复杂类型的数据,因为基本类型数据的定义都会重新开辟新的内存. 浅拷贝拷贝的是内存地址,只是增加一个指针指向已有的内存,这时多个数据共用一个内存空间:深拷贝是新增了指针并且新开辟了内存空间,新指针指向新的内存. 浅拷贝: var a={ name:'WangJing', sex:'女', age:'25

C++拷贝构造函数(深拷贝,浅拷贝)

http://www.cnblogs.com/BlueTzar/articles/1223313.html 对于普通类型的对象来说,它们之间的复制是很简单的,例如:int a=88;int b=a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量.下面看一个类对象拷贝的简单例子. 1 #include <iostream> 2 using namespace std; 3 4 class CExample { 5 private: 6 int a; 7 public: 8

python 中 深拷贝和浅拷贝的理解

在总结 python 对象和引用的时候,想到其实 对于python的深拷贝和浅拷贝也可以很好对其的进行理解. 在python中,对象的赋值的其实就是对象的引用.也就是说,当创建一个对象,然后赋给另外一个变量之后,实际上只是拷贝了这个对象的引用. 我们先用  利用切片操作和工厂方法list方法 来阐述一下浅拷贝. 举个栗子: Tom = ['Tom', ['age', 10]] Jack = Tom[:] ……切片操作 June = list(Tom) 接下来查看一下 上述三个变量的引用: >>

实现CMyString类--深拷贝与浅拷贝

实现CMyString类,实现赋值运算符函数(深拷贝与浅拷贝(写时拷贝)) 深拷贝:赋值运算符函数说明 步骤1.释放原来的内存空间 步骤2.再重新开辟要赋值的对象的大小的空间 步骤3.再将另一个对象的值拷贝给this对象 友元函数的说明:由于输出重载需要俩个参数,不能再隐含this指针,故使用友元函数 代码如下: <span style="font-size:18px;">class CMyString { public: CMyString() :str(new char

python 深拷贝和浅拷贝之可变和不可变对象总结

了解深拷贝和浅拷贝之前先要理解可变与不可变对象 python只允许使用引用传递,有可变对象和不可变对象,可变对象:list,dict.不可变对象有:int,string,float,tuple Python int,string,float,tuple不可变举栗子: def int_object(): i = 89 j = 89 print(id(89)) print('i id:' + str(id(i))) print('j id:' + str(id(j))) print(i is j)

深拷贝与浅拷贝

一.浅拷贝和深拷贝 所谓浅拷贝,就是由默认的拷贝构造函数所实现的对数据成员逐一赋值.若类中含有指针类型的数据,这种方式只是简单的把指针的指向赋值给新成员,但并没有给新成员分配内存,因此这种方式必然会导致错误.为了解决浅拷贝出现的错误,必须显示的定义一个拷贝构造函数,使之不但复制数据成员,而且为对象分配各自的内存空间,这就是所谓的深拷贝. 二.浅拷贝 浅拷贝就是由默认的拷贝构造函数所实现的数据成员逐一赋值.通常默认的拷贝构造函数能够胜任这个工作,但是若类中含有指针类型的数据,这种数据成员逐一赋值的

【转】 C++的深拷贝与浅拷贝

对于普通类型的对象来说,它们之间的复制是很简单的,例如:int a=88;int b=a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量.下面看一个类对象拷贝的简单例子. 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 #include <iostream> using namespace std; class CExample { private:      int a; public:      CE

python 深拷贝与浅拷贝

浅拷贝的方式有: lst=[1,2,3] (1)直接赋值: lst_cp = lst (2)for循环遍历生成:lst_cp= [i for i in lst] (3)copy模块下,copy.copy仍为浅拷贝 深拷贝的方式 (1)借助copy模块 >>> import copy >>> lst_cp = copy.deepcopy(lst) 以上方法的测试: 注意:因为string类型是不可变类型,所以修改string元素时会新创建一个地址空间放置数据 (1)直接赋