美文网首页
字符串用copy修饰

字符串用copy修饰

作者: Hyman0819 | 来源:发表于2016-10-18 19:38 被阅读25次

属性修饰符:

    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确实没有受到影响

原因:
将之前的可变字符串变为不可变字符串,因为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比较好

相关文章

网友评论

      本文标题:字符串用copy修饰

      本文链接:https://www.haomeiwen.com/subject/bzytuttx.html