浅拷贝:只创建一个新的指针,指向原指针指向的内存。
深拷贝 : 创建一个新的指针,并开辟新的内存空间,内容拷贝自原指针指向的内存,并指向它。
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