源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

详解iOS的深浅拷贝

  • 时间:2021-10-07 12:09 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:详解iOS的深浅拷贝
[b]前言[/b] [b]OC对象的三种拷贝方式[/b] OC的对象拷贝有如下三种方式,很多时候我们把深复制和完全复制混为一谈,其他他们是有区别的,具体如下 [b]浅复制(shallow copy):[/b]在浅复制操作时,对于被复制对象的每一层都是指针复制。 [b]深复制(one-level-deep copy):[/b]在深复制操作时,对于被复制对象,至少有一层是深复制。 [b]完全复制(real-deep copy):[/b]在完全复制操作时,对于被复制对象的每一层都是对象复制。 [b]两图以避之[/b] [img]http://files.jb51.net/file_images/article/201608/201682112425868.png?201672112438[/img] [img]http://files.jb51.net/file_images/article/201608/201682112522837.png?201672112531[/img] [b]理解深复制(mutableCopy)[/b] 浅复制很简单,就不演示了,看上面的图就懂了,只是简单的指针拷贝,所以改变原对象或者拷贝后的对象,都会影响另外一个对象。 从上图我们可以看到[code]mutableCopy[/code]对于任何对象都是内容复制,也就是说进行了深复制。 [b]上代码:[/b]
  NSMutableArray * dataArray1=[NSMutableArray arrayWithObjects:
                [NSMutableString stringWithString:@"1"],
                [NSMutableString stringWithString:@"2"],
                [NSMutableString stringWithString:@"3"],
                [NSMutableString stringWithString:@"4"],
                nil

                ];
  NSMutableArray * dataArray2=[NSMutableArray arrayWithObjects:
                [NSMutableString stringWithString:@"one"],
                [NSMutableString stringWithString:@"two"],
                [NSMutableString stringWithString:@"three"],
                [NSMutableString stringWithString:@"four"],
                dataArray1,
                nil
                ];

  NSMutableArray * dataArray3;
  NSMutableString * mStr;

  dataArray3=[dataArray2 mutableCopy];

  mStr = dataArray2[0];
  [mStr appendString:@"--ONE"];

  NSLog(@"dataArray3:%@",dataArray3);
  NSLog(@"dataArray2:%@",dataArray2);
[b]输出如下:[/b]
2016-07-31 17:40:30.702 test1[2113:169774] dataArray3:(
  "one--ONE",
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)
2016-07-31 17:40:30.703 test1[2113:169774] dataArray2:(
  "one--ONE",
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)
看上面的输出,我们发现我们改变原数组[code]dataArray2[/code],竟然也会影响深复制后的[code]dataArray3[/code],不是说好的内容复制吗,为什么会这样? 这里我们来说说[b]深复制和完全复制的区别[/b]。 我们知道深复制,就是把原有对象的内容直接克隆一份到新对象,但是这里有一个坑就是他只会复制一层对象,而不会复制第二层甚至更深层次的对象。 代码[code]dataArray3=[dataArray2 mutableCopy];[/code]只是对数组[code]dataArray2[/code]本身进行了内容拷贝,但是里面的字符串对象却没有进行内容拷贝,而是进行的浅复制,那么[code]dataArray2[/code]和[code]dataArray3[/code]里面的对象是共享同一份的。所以才会出现上面的情况。 [b]单层深复制[/b] 那么如何解决上面的问题呢? 可以使用如下代码
  dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];
输出如下:
2016-07-31 17:45:48.472 test1[2151:173221] dataArray3:(
  one,
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)
2016-07-31 17:45:48.472 test1[2151:173221] dataArray2:(
  "one--ONE",
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)
可以看到[code]dataArray3[/code]并没有被改变,但是别高兴的太早,我们再来改改。 代码如下:
  NSMutableArray * dataArray1=[NSMutableArray arrayWithObjects:
                [NSMutableString stringWithString:@"1"],
                [NSMutableString stringWithString:@"2"],
                [NSMutableString stringWithString:@"3"],
                [NSMutableString stringWithString:@"4"],
                nil

                ];
  NSMutableArray * dataArray2=[NSMutableArray arrayWithObjects:
                [NSMutableString stringWithString:@"one"],
                [NSMutableString stringWithString:@"two"],
                [NSMutableString stringWithString:@"three"],
                [NSMutableString stringWithString:@"four"],
                dataArray1,
                nil
                ];

  NSMutableArray * dataArray3;
  NSMutableString * mStr;

  dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];

  NSMutableArray *mArr = (NSMutableArray *)dataArray2[4];
  mStr = mArr[0];
  [mStr appendString:@"--ONE"];

  NSLog(@"dataArray3:%@",dataArray3);
  NSLog(@"dataArray2:%@",dataArray2);
输出如下:
2016-07-31 17:47:19.421 test1[2174:174714] dataArray3:(
  one,
  two,
  three,
  four,
    (
    "1--ONE",
    2,
    3,
    4
  )
)
2016-07-31 17:47:19.421 test1[2174:174714] dataArray2:(
  one,
  two,
  three,
  four,
    (
    "1--ONE",
    2,
    3,
    4
  )
)
可以看到深复制又失效了,这是因为[code]dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];[/code]仅仅能进行一层深复制,对于第二层或者更多层的就无效了,那怎么办呢? 别急,我们还有大招没放。 [b]完全复制[/b] 要想对多层集合对象进行复制,我们需要进行完全复制,这里可以使用归档和接档。 实现代码如下:
  dataArray3 = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:dataArray2]];
此时输出如下:
2016-07-31 17:49:55.561 test1[2202:177163] dataArray3:(
  one,
  two,
  three,
  four,
    (
    1,
    2,
    3,
    4
  )
)
2016-07-31 17:49:55.562 test1[2202:177163] dataArray2:(
  one,
  two,
  three,
  four,
    (
    "1--ONE",
    2,
    3,
    4
  )
)
可以看到[code]dataArray3[/code]没有被[code]dataArray2[/code]的修改影响。 [b]类复制[/b] 说完了对象的复制,我们来看看如何实现类的复制,因为比较简单,直接放上代码 [b]定义类复制[/b]
#import <Foundation/Foundation.h>
@interface Person : NSObject<NSCopying>
@property(strong,nonatomic)NSString *age;
@property(strong,nonatomic)NSString *name;
@end
#import "Person.h"
@implementation Person
- (id)copyWithZone:(NSZone *)zone
{
  Person *person = [[Person allocWithZone:zone] init];
  person.age = self.age;
  person.name = self.name;
  return person;
}
@end
[b]调用[/b]
  Person *person = [[Person alloc]init];
    person.age = @"dsdsd";
    person.name = @"dsdsdddww";

    Person *copyPerson = [person copy];
    NSLog(@"%@-----%@",copyPerson.age, copyPerson.name);
可以看到[code]copyPerson[/code]的两个属性和[code]persona[/code]一样。 [b]@property中的copy关键字[/b] 在设置NSString类型的属性的时候,我们最好设置为copy类型,这样别人使用我们定义的属性的时候,他不管怎么改动该属性的赋值,都不会影响我们给该属性赋的值,为什么呢? 下面我们来看看 [img]http://files.jb51.net/file_images/article/201608/201682113003665.png?201672113013[/img] 如上图所示,[code]string2[/code]的属性是[code]copy[/code]类型,可以看到是无法被修改的。 因为此时[code]string2[/code]和[code]copystring[/code]的内存地址不一样,修改一个,不会影响另外一个。 [img]http://files.jb51.net/file_images/article/201608/201682113056147.png?20167211318[/img] 上图所示,如果[code]string2[/code]的属性是[code]strong[/code]类型,就可以被修改,如下图所示: 因为此时[code]string2[/code]和[code]copystring[/code]的内存地址都是一样的,修改一个,两个就同时被修改 [b]copy关键字的NSMutableString崩溃[/b] [img]http://files.jb51.net/file_images/article/201608/201682113142351.png?201672113152[/img] [b]原因:[/b] [code]copy[/code]关键字的[code]string[/code]的[code]setter[/code]方法实际上是把参数copy之后再赋值给变量[code]_string[/code],那么此时变量[code]_string[/code]虽然被申明为[code]NSMutableString[/code],但是[code]copy[/code]之后,就把 变量[code]_string[/code]变成了不可变的[code]NSString[/code]类型,所以就会出现方法报错,提示对不可变的[code]NSString[/code]使用了[code]NSMutableString[/code]的方法[code]appendString[/code]。 [b]总结[/b] 以上就是iOS的深浅拷贝的详细内容,希望本文在大家开发iOS的过程中能有所帮助。
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部