copy、mutableCopy与浅拷贝、深拷贝

2018-11-27  本文已影响0人  SimpleSJ

1.先来看一段代码

    NSArray *array;
    NSMutableArray *arrM = [NSMutableArray arrayWithObjects:@"a",@"b",@"c", nil];
    array = arrM;
    NSLog(@"-----%@",array);
    [arrM addObject:@"d"];
    NSLog(@"=======%@",array);
//输出结果: 
-----(
    a,
    b,
    c
)
 =======(
    a,
    b,
    c,
    d
)

思考:arrM先赋值给array,再增加d元素,为什么array中却也加入了新的元素?
这里先分析造成这个情况的原因如下图:

问题分析一
//造成这种问题的关键代码
//array = arr;
array = [arr copy];

copy方法做了那些事情?起到了什么样的作用,下面我们仔细研究一下.

2.copy、mutableCopy

copy、mutableCopy的本质:修改原来的对象不能影响到拷贝出来得对象;修改拷贝出来的对象也不能影响到原来的对象,总之就是两个对象互不影响.

    NSMutableArray *arrM1 = [NSMutableArray array];
    NSMutableArray *arrM2 = [NSMutableArray arrayWithObjects:@"1",@"2", nil];
    arrM1 = [arrM2 mutableCopy];
    [arrM1 addObject:@"aa"];
    [arrM2 addObject:@"bb"];
    NSLog(@"arrM1 = %@,arrM2 = %@,arrM1内存地址 = %p,arrM2内存地址 = %p",arrM1,arrM2,arrM1,arrM2);
//输出结果:
arrM1 = (
    1,
    2,
    aa
),arrM2 = (
    1,
    2,
    bb
),arrM1内存地址 = 0x600000242130,arrM2内存地址 = 0x6000002418c0
    NSMutableArray *arrM = [NSMutableArray array];
    NSArray *dataArray = @[@"1",@"2"];
    arrM = [dataArray mutableCopy];
    [arrM addObject:@"3"];
    NSLog(@"arrM = %@,dataArray = %@,arrM内存地址 = %p,dataArray内存地址 = %p",arrM,dataArray,arrM,dataArray);
//输出:
arrM = (
    1,
    2,
    3
),dataArray = (
    1,
    2
),arrM内存地址 = 0x600000055cc0,dataArray内存地址 = 0x600000222b20
    NSArray *dataArray;
    NSMutableArray *arrM = [NSMutableArray arrayWithObjects:@"1",@"2", nil];
    dataArray = [arrM copy];
    [arrM addObject:@"3"];
    NSLog(@"dataArray = %@,arrM = %@,dataArray内存地址 = %p,arrM内存地址 = %p",dataArray,arrM,dataArray,arrM);
//输出:
dataArray = (
    1,
    2
),arrM = (
    1,
    2,
    3
),dataArray内存地址 = 0x604000220ea0,arrM内存地址 = 0x604000244320
    NSArray *arr1;
    NSArray *arr2 = @[@"1",@"2",@"3"];
    arr1 = [arr2 copy];
    NSLog(@"arr1 = %@,arr2 = %@,arr1内存地址 = %p,arr2内存地址 = %p",arr1,arr2,arr1,arr2);
//输出:
arr1 = (
    1,
    2,
    3
),arr2 = (
    1,
    2,
    3
),arr1内存地址 = 0x60000045df10,arr2内存地址 = 0x60000045df10
例四图解

前三个例子都会生成一个新的对象,保证两个对象之间进行修改,互不影响,但是例四在不生成新对象的情况下,依然能保证两个对象之间进行修改,互不影响.
原因: 因为原来的对象是不能修改的, 拷贝出来的对象也是不能修改的,既然两个都不能修改, 所以永远不能影响到另外一个对象, 那么就满足了我们上面提到的copy、mutableCopy的本质,

3.浅拷贝、深拷贝

在调用了copy、mutableCopy方法之后:

4.补充

NSArray *array;
    NSMutableArray *arrM = [NSMutableArray arrayWithObjects:@"a",@"b",@"c", nil];
    array = arrM;
    NSLog(@"-----%@",array);
    [arrM addObject:@"d"];
    NSLog(@"=======%@",array);
//输出结果: 
-----(
    a,
    b,
    c
)
 =======(
    a,
    b,
    c,
    d
)

这里出现这种情况的原因,上面也说过了就是array = arrM;就是这句代码造成的.而且其效果与浅拷贝相同,就是拷贝了一个指针指向了数组对象.
所以:=是浅拷贝,大多数情况下应该是浅拷贝吧.深拷贝的情况没有注意过,如果写的不对,请指证,我及时修改.3Q

5.类比Foundation框架中的其他数据类型

上一篇下一篇

猜你喜欢

热点阅读