iOS深拷贝和浅拷贝

  • 浅拷贝:只创建一个新的指针,指向原指针指向的内存。

  • 深拷贝 : 创建一个新的指针,并开辟新的内存空间,内容拷贝自原指针指向的内存,并指向它。

1. 非集合对象的copy与mutableCopy
1.1 NSString 探究

-(void)stringDemo{
    NSString *string = @"str1";
    //copy返回的是不可变对象,str2不能被修改,因此会发生崩溃
    NSString *copyString = [string copy];
    NSMutableString *mutableCopyString = [string mutableCopy];
    NSLog(@"string: %p",string);
    NSLog(@"copyString: %p",copyString);
    NSLog(@"mutableCopyString: %p",mutableCopyString);
}
//  控制台输出
2020-06-18 15:11:34.933475+0800 深浅拷贝探究[4231:359229] string: 0x102be2020
2020-06-18 15:11:34.933590+0800 深浅拷贝探究[4231:359229] copyString: 0x102be2020
2020-06-18 15:11:34.933671+0800 深浅拷贝探究[4231:359229] mutableCopyString: 0x600000f4b150

1.2 NSMutableString 探究

-(void)mutableStringDemo{
    NSMutableString *string = [NSMutableString stringWithString:@"test002"];
    NSMutableString *copyString = [string copy];
    //copy返回的是不可变对象,mstr2不能被修改,因此会发生崩溃
    //[mstr2 appendString:@"test"];
    NSMutableString *mutableCopyString = [string mutableCopy];
    //[mstr3 appendString:@"modify"];
    NSLog(@"string: %p",string);
    NSLog(@"copyString: %p",copyString);
    NSLog(@"mutableCopyString: %p",mutableCopyString);
}
//  控制台输出
2020-06-18 15:16:09.342807+0800 深浅拷贝探究[4291:363596] string: 0x60000333b240
2020-06-18 15:16:09.342907+0800 深浅拷贝探究[4291:363596] copyString: 0x82604a6aef771f79
2020-06-18 15:16:09.342984+0800 深浅拷贝探究[4291:363596] mutableCopyString: 0x60000333afa0

结论:

  • 非容器对象使用mutableCopy都是深拷贝
  • 非容器对象中不可变类型使用copy是浅拷贝,可变类型使用copy是深拷贝
  • 非容器对象copy得到的类型一定是不可变的;mutableCopy得到的类型一定是可变的

2. 集合对象的copy与mutableCopy
2.1 NSArray 探究

-(void)arrayDemo{
    NSArray *arry = [[NSArray alloc] initWithObjects:@"value1", @"value2",nil];
    NSArray *copyArry = [arry copy];
    NSArray *mutableCopyArry = [arry mutableCopy];
    NSLog(@"arry: %p",arry);
    NSLog(@"copyArry: %p",copyArry);
    NSLog(@"mutableCopyArry: %p",mutableCopyArry);
}
//  控制台输出
2020-06-18 15:20:09.696461+0800 深浅拷贝探究[4314:366430] arry: 0x60000324e120
2020-06-18 15:20:09.696630+0800 深浅拷贝探究[4314:366430] copyArry: 0x60000324e120
2020-06-18 15:20:09.696711+0800 深浅拷贝探究[4314:366430] mutableCopyArry: 0x600003c43000

2.2 NSMutableArray 探究

-(void)mutableArrayDemo{
    NSMutableArray *arry = [[NSMutableArray alloc] initWithObjects:@"value1", @"value2",nil];
    NSMutableArray *copyArry = [arry copy];
    //copy返回的是不可变对象,marry2不能被修改,因此会崩溃
    //[arry2 addObject:@"value3"];
    NSMutableArray *mutableCopyArry = [arry mutableCopy];
    NSLog(@"arry: %p",arry);
    NSLog(@"copyArry: %p",copyArry);
    NSLog(@"mutableCopyArry: %p",mutableCopyArry);
}
//  控制台输出
2020-06-18 15:22:00.245880+0800 深浅拷贝探究[4332:368245] arry: 0x6000002c2190
2020-06-18 15:22:00.246018+0800 深浅拷贝探究[4332:368245] copyArry: 0x600000cc0e00
2020-06-18 15:22:00.246112+0800 深浅拷贝探究[4332:368245] mutableCopyArry: 0x6000002c2160

总结
*容器对象(NSArray,NAMutableArray;NSDictionary,NSMutableDictionary;NSSet集合)遵循非容器对象的拷贝原则
*容器内的元素都是浅拷贝

3.自定义对象

自定义对象使用copy和mutableCopy需要遵守NSCopying和NSMutableCopying协议,


- (id)copyWithZone:(nullable NSZone *)zone;

- (id)mutableCopyWithZone:(nullable NSZone *)zone;

4. 应用
4. 1 在面试的时候面试官会经常问到NSString属性是用copy还是用strong修饰,如果用strong会有什么样的问题?

  • 我们可以看到我们定义的NSString类型的属性用strong修饰,当我们给它传一个NSMutableString类型的数据时,它的指针指向了该数据,之后会随NSMutableString的特性去改变;
  • 这样就违背了我们定义NSString类型的初衷了,在项目里面谨慎使用;
  • 如果使用copy就不会有这样的问题,我们定义NSString类型,使用copy,不管给它传的值是NSString还是NSMutableString最后都会得到NSString类型;

4. 2 在面试的时候面试官会经常问到NSString属性是用copy还是用strong修饰,如果用strong会有什么样的问题?

  • 我们可以看到我们定义的NSString类型的属性用strong修饰,当我们给它传一个NSMutableString类型的数据时,它的指针指向了该数据,之后会随NSMutableString的特性去改变
  • 这样就违背了我们定义NSString类型的初衷了,在项目里面谨慎使用
  • 如果使用copy就不会有这样的问题,我们定义NSString类型,使用copy,不管给它传的值是NSString还是NSMutableString最后都会得到NSString类型

所以,NSMutableString类型使用strong修饰,NSString类型最好使用copy,谨慎使用strong

总结.png

参考
iOS深拷贝和浅拷贝_作者:孤单的寸衣
深拷贝和浅拷贝解析_作者:Devbrave

你可能感兴趣的:(iOS深拷贝和浅拷贝)