- (void)test
{
NSMutableString *mStr = [NSMutableString stringWithFormat:@"abc"];
self.rStr = mStr;
self.cStr = mStr;
NSLog(@"mStr:%p,%p", mStr,&mStr);
NSLog(@"retainStr:%p,%p", _rStr, &_rStr);
NSLog(@"copyStr:%p,%p", _cStr, &_cStr);
}
2017-09-05 14:58:27.080 StrTest[5342:182169] mStr:0x6000000725c0,0x7fff4fc01a28
2017-09-05 14:58:27.080 StrTest[5342:182169] retainStr:0x6000000725c0,0x7fbb2d70d698
2017-09-05 14:58:27.080 StrTest[5342:182169] copyStr:0xa000000006362613,0x7fbb2d70d6a0
可以看出:
1、mStr和rStr指向同样的地址0x6000000725c0,他们指向的是同一个对象@“abc”,而mStr变量自身在内存中的地址为0x7fff4fc01a28,rStr变量自身在内存中的地址为0x7fbb2d70d698,说明retain是复制指针,但是复制后两个指针指向的是同一个对象。
2、cStr与mStr变量自身在内存中的地址和指针指向的地址都是不一样的,因为他们指向的是不同的对象,所以copy在这里是深复制,复制了一个新的对象,同时又有一个新的指针指向新的对象。
然后:
[mStr appendString:@"de"];
NSLog(@"retainStr:%@", _rStr);
NSLog(@"copyStr:%@", _cStr);
2017-09-05 15:46:03.397 StrTest[5407:257975] retainStr:abcde
2017-09-05 15:46:03.397 StrTest[5407:257975] copyStr:abc
- 结论:如果不希望字串的值跟着mStr变化,就用copy来设置string的属性;如果希望字串的值跟着赋值的字串的值变化而变化,就使用strong、retain。
注意:上面的情况是针对于当把NSMutableString赋值给NSString的时候,才会有不同,如果是赋值是NSString对象,那么使用copy还是strong,结果都是一样的,因为NSString对象根本就不能改变自身的值,它是不可变的。
总结下,对源头是NSMutableString的字符串,retain仅仅是指针引用,增加了引用计数器,这样源头改变的时候,用这种retain方式声明的变量(无论被赋值的变量是可变的还是不可变的),它也会跟着改变;而copy声明的变量,它不会跟着源头改变,它实际上是深拷贝。
对源头是NSString的字符串,无论是retain声明的变量还是copy声明的变量,当第二次源头的字符串重新指向其它的地方的时候,它还是指向原来的最初的那个位置,也就是说其实两者都是指针引用,也就是浅拷贝。