NSString使用之Copy和Strong

来源:互联网 发布:算法设计与分析 沙特 编辑:程序博客网 时间:2024/05/22 12:53

背景:

       前些天去面试了,面试官很牛X,目测大牛,问了很多问题。我记得第一个好像是用assign修饰对象??后面又衍生出了strong和copy修饰字符串的区别, 搬砖搬了多年,只记得这些修饰词,大概干嘛干嘛用的了。这些时间的面试,也通过这么多年的搬砖,越来越意识到了自己该做点什么了。好了闲聊到这里,以下内容是我根据其他博主摘抄学习而来, 好了直接上代码。

主题:

///声明@interface ViewController ()@property (nonatomic, strong) NSString *strongString;@property (nonatomic, copy) NSString *copyedString;@end

然后我们来给这两个属性赋值,这里分为两种情况

  1. 用不变字符串(NSString)来给它们赋值;

  2. 用可变字符串(NSMutableString)赋值;


不变字符串(NSString)来给它们赋值

     NSString *string = [NSString stringWithFormat:@"abc"];     self.strongString = string;     self.copyedString = string;     NSLog(@"origin string: %p, %p, %@", string, &string, string);     NSLog(@"strong string: %p, %p, %@", _strongString, &_strongString, _strongString);     NSLog(@"copy   string: %p, %p, %@", _copyedString, &_copyedString, _copyedString);      /*       在这里我是忘记了&取地址符,我记录下,与主题无关         1、a是一个数组,代表一个数组第一个元素的地址            &a操作是可以的,实际上&a和a得到的地址是一个地址,只是存储单元不一样            a 指的是第一个元素的地址  &a则表示这个数组的地址         2、直接p 访问得到的是p这个变量存储的地址(首地址),&p则取了p这个变量的地址(变量自身在内存中的地址)        */

打印结果:

2017-04-02  Text1[1226:55079] origin string: 0xa000000006362613, 0x7fff5f96c658, abc2017-04-02  Text1[1226:55079] strong string: 0xa000000006362613, 0x7fc3e48033e8, abc2017-04-02  Text1[1226:55079] copy   string: 0xa000000006362613, 0x7fc3e48033f0, abc

这个结果我们看到 ,用 不可变字符串(NSString)给其赋值 不管是strong还是copy属性的对象,其指向的地址都是同一个,即为 NSString对象 指向的地址。


可变字符串(NSMutableString)给其赋值

    NSMutableString *string = [NSMutableString stringWithFormat:@"abc"];    self.strongString = string;    self.copyedString = string;    NSLog(@"origin string: %p, %p, %@", string, &string, string);    NSLog(@"strong string: %p, %p, %@", _strongString, &_strongString, _strongString);    NSLog(@"copy   string: %p, %p, %@", _copyedString, &_copyedString, _copyedString);    ///第二步    [string appendString:@"edf"];    [(NSMutableString *)_strongString appendString:@"ghjk"];    NSLog(@"origin string: %p, %p, %@", string, &string, string);    NSLog(@"strong string: %p, %p, %@", _strongString, &_strongString, _strongString);    NSLog(@"copy   string: %p, %p, %@", _copyedString, &_copyedString, _copyedString);

控制台结果:

Text1[1263:58830] origin string: 0x61000026ad00, 0x7fff5dfde658, abcText1[1263:58830] strong string: 0x61000026ad00, 0x7fb78ad04cd8, abcText1[1263:58830] copy   string: 0xa000000006362613, 0x7fb78ad04ce0, abc Text1[1263:58830] origin string: 0x61000026ad00, 0x7fff5dfde658, abcedfghjk Text1[1263:58830] strong string: 0x61000026ad00, 0x7fb78ad04cd8, abcedfghjkText1[1263:58830] copy   string: 0xa000000006362613, 0x7fb78ad04ce0, abc

相信大家已经看到, 此时copy属性字符串已不再指向string字符串对象,而是深拷贝了string字符串,并让_copyedString对象指向这个字符串。

这个时候我们根据第二步,我们如果去修改string字符串的话,可以看到:因为_strongString与string是指向同一对象,所以_strongString的值也会跟随着改变(需要注意的是,此时_strongString的类型实际上是NSMutableString,而不是NSString);而_copyedString是指向另一个对象的,所以并不会改变。


结论

由于NSMutableString是NSString的子类,所以一个NSString指针可以指向NSMutableString对象,让我们的strongString指针指向一个可变字符串是OK的。

而上面的例子可以看出,当源字符串是NSString时,由于字符串是不可变的,所以,不管是strong还是copy属性的对象,都是指向源对象,copy操作只是做了次浅拷贝。

当源字符串是NSMutableString时,strong属性只是增加了源字符串的引用计数,而copy属性则是对源字符串做了次深拷贝,产生一个新的对象,且copy属性对象指向这个新的对象。另外需要注意的是,这个copy属性对象的类型始终是NSString,而不是NSMutableString,因此其是不可变的。

这里还有一个性能问题,即在源字符串是NSMutableString,strong是单纯的增加对象的引用计数,而copy操作是执行了一次深拷贝,所以性能上会有所差异。而如果源字符串是NSString时,则没有这个问题。

所以,在声明NSString属性时,到底是选择strong还是copy,可以根据实际情况来定。不过,一般我们将对象声明为NSString时,都不希望它改变,所以大多数情况下,我们建议用copy,以免因可变字符串的修改导致的一些非预期问题。


再说下 assign 修饰对象,我也是查资料,如果有更好认知,请斧正, 谢谢!

///声明@interface ViewController ()@property (nonatomic,weak)   NSDate *weakDate;@property (nonatomic,assign) NSDate *assignDate;@property (nonatomic,strong) NSDate *strongDate;@end
///实现    _strongDate = [NSDate date];    _weakDate = _strongDate;    _assignDate = _strongDate;    NSLog(@"%p, %p, %@", _strongDate, &_strongDate, _strongDate);    NSLog(@"%p, %p, %@", _weakDate, &_weakDate, _weakDate);    NSLog(@"%p, %p, %@", _assignDate, &_assignDate, _assignDate);    _strongDate = nil;    NSLog(@"strong属性: %@", _strongString);    NSLog(@"weak属性: %@", _weakDate);    NSLog(@"assign属性: %@", _assignDate);
///打印结果Text1[1383:70114] 0x60800000c110, 0x7ff970e05338, 2017-04-02 02:33:16 +0000Text1[1383:70114] 0x60800000c110, 0x7ff970e05328, 2017-04-02 02:33:16 +0000Text1[1383:70114] strong属性:(null)Text1[1383:70114] weak属性:(null)(lldb)  

结论:

weak 修饰对象时候,当对象被释放掉后,指针会指向 nil
strong 修饰对象时候,当对象被释掉后,指针会指向 nil

assign和weak区别

      weak特性要求不保留传入的对象。如果该对象被释放,那么相应的实例变量会被自动赋为nil。这么做可以避免产生悬空指针。悬空指针指向的是不再存在的对象。向悬空指针发送消息通常会导致程序崩溃。相应的存方法会将传入的对象直接赋给实例变量。

0 0
原创粉丝点击