字符串为什么用copy修饰

属性修饰符:

    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的后果

1.使strong修饰一个NSString对象
2.在viewDidLoad中实例化一个空的可变字符串
3.给可变字符串添加内容
4.让strong修饰的NSString对象指向可变字符串对象
5.继续向可变字符串中添加内容
6.打印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确实没有受到影响

原因:
将之前的可变字符串变为不可变字符串,因为NSString不支持append添加操作,我这里的两次str赋值操作,其实是让str重新指向了一片内存空间,并不是修改了str原本内存中的值
(OC中对象即指针,实际上存储的是内存地址,self.name = str; 实际是将str存储的@"xiaoming"这块地址给了self.name, self.name还指向着@"xiaoming")
所以改变str的指向后, self.name的指向并没有改变,输出没有受到影响

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容