属性修饰符:
MRC:
assign:基本数据类型(当出现循环引用时,也要用assign)
retain:除Block和NSString外的其他对象
copy:一般用于NSString和Block
ARC:
strong:默认
weak:多用于ui和解决循环引用
copy:用于NSString和Block
assign:非OC对象
既然NSString属于OC对象,那么我们先不使用Copy修饰,在ARC模式下,声明的属性默认是strong修饰,接下来就演示strong修饰NSString的后果
- 使strong修饰一个NSString对象
- 在viewDidLoad中实例化一个空的可变字符串
- 给可变字符串添加内容
- 让strong修饰的NSString对象指向可变字符串对象
- 继续向可变字符串中添加内容
- 打印strong修饰的NSString对象结果
#import "ViewController.h"
@interface ViewController ()
/* 为什么字符串使用copy修饰?
这里先不使用copy修饰,NSString是OC对象,并且是ARC模式,所以先使用strong来修饰演示
*/
@property (nonatomic,strong) NSString *name;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSMutableString *str = [NSMutableString string];
[str appendString:@"xiaoming"];
self.name = str;
[str appendString:@"xiaogang"];
NSLog(@"%@",_name);
}
@end
结果为:
2016-07-31 13:51:35.218 字符串使用copy修饰[9376:241671] xiaomingxiaogang
结果分析:
如果使用strong修复NSString类型属性
self.name 指向可变字符串对象的地址
当可变字符串内容发生变化时,self.name相对应的也发生变化
使用copy修饰后,将可变字符串重新拷贝一份,重新开辟内存空间,修改mutableString的值,不会对self.name造成影响
2016-07-31 14:22:00.070 字符串使用copy修饰[9548:269772] str:0x7fa871488750,name:0xa006412031812da8
2016-07-31 14:22:00.074 字符串使用copy修饰[9548:269772] str:0x7fa871488750,name:0xa006412031812da8
2016-07-31 14:22:00.074 字符串使用copy修饰[9548:269772] xiaoming
刚刚的演示,我使用了NSMutableString(可变字符串),对mutableString执行copy操作,属于深拷贝,所以开辟的新内存空间,如果使用的是NSString(不可变内存),对NSString进行copy属于浅拷贝,不会开辟新的内存空间,是不是就不会出现这个问题了呢?
接下来使用NSString来进行演示,将属性修饰符再次改回strong
#import "ViewController.h"
@interface ViewController ()
/* 为什么字符串使用copy修饰?
这里先不使用copy修饰,NSString是OC对象,并且是ARC模式,所以先使用strong来修饰演示
*/
@property (nonatomic,strong) NSString *name;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSString *str = [NSString string];
str = @"xiaoming";
self.name = str;
NSLog(@"str:%p,name:%p",str,_name);
str = @"xiaogang";
NSLog(@"str:%p,name:%p",str,_name);
NSLog(@"%@",_name);
}
@end
打印结果:
2016-07-31 14:17:19.127 字符串使用copy修饰[9499:264651] str:0x10c937060,name:0x10c937060
2016-07-31 14:17:19.128 字符串使用copy修饰[9499:264651] str:0x10c9370a0,name:0x10c937060
2016-07-31 14:17:19.128 字符串使用copy修饰[9499:264651] xiaoming
从结果上看,重新设置str的值后,self.name确实没有受到影响
原因:
将之前的可变字符串变为不可变字符串,因为NSMutableString不支持append添加操作,我这里的两次str赋值操作,其实是让str重新指向了一片内存空间,并不是修改了str原本内存中的值
(OC中对象即指针,实际上存储的是内存地址,self.name = str; 实际是将str存储的@"xiaoming"这块地址给了self.name,self.name还指向着@"xiaoming")
所以改变str的指向后,self.name的指向并没有改变,输出没有受到影响
结论
你只需要记住一点,当你给你的的NSString对象赋值时,如果来源是NSMutableString,那么这种情况就必须要用copy;如果你确定来源是不可变类型的,比如@"http://www.jianshu.com/users/691d9ed740cf/latest_articles"
这种固定的字符串,那么用strong比较好