NSString内存小结,以及isEqual与isEqualToString的研究

思考

一言不合先上个小小的思考吧

NSString *str0 = @"1";
NSString *str1 = [NSString stringWithFormat:@"1"];//
    
NSLog(@"%d", str0 == str1);
NSLog(@"%d", [str0 isEqualToString:str1]);
NSLog(@"%d", [str0 isEqual:str1]);

大家可以短暂的思索下输出的结果,后面我们再说

关于NSString内存

带着测试来看吧

NSString *str0 = @"1";
NSString *str1 = [NSString stringWithFormat:@"1"];//
NSString *str2 = [[NSString alloc] initWithString:@"1"];//
NSMutableString *str3 = [[NSMutableString alloc] initWithString:@"1"];//
NSString *strs0copy = str0.copy;
NSString *str3copy = str3.copy;
NSString *str0mucopy = str0.mutableCopy;
NSString *str00mucopy = str0.mutableCopy;

NSLog(@"str0:%p", str0);
NSLog(@"str1:%p", str1);
NSLog(@"str2:%p", str2);
NSLog(@"strs0copy:%p", strs0copy);
NSLog(@"str3:%p", str3);
NSLog(@"str3copy:%p", str3copy);
NSLog(@"str0mucopy:%p", str0mucopy);
NSLog(@"str00mucopy:%p", str00mucopy);
    

我们看下输出的结果以及堆栈信息

2017-06-08 15:50:52.198 NSString_ memory[68403:4037182] str0:0x103513068
2017-06-08 15:50:52.199 NSString_ memory[68403:4037182] str1:0xa000000000000311
2017-06-08 15:50:52.199 NSString_ memory[68403:4037182] str2:0x103513068
2017-06-08 15:50:52.199 NSString_ memory[68403:4037182] strs0copy:0x103513068
2017-06-08 15:50:52.199 NSString_ memory[68403:4037182] str3:0x618000263c00
2017-06-08 15:50:52.199 NSString_ memory[68403:4037182] str3copy:0xa000000000000311
2017-06-08 15:50:52.199 NSString_ memory[68403:4037182] str0mucopy:0x618000263bc0
2017-06-08 15:50:52.200 NSString_ memory[68403:4037182] str00mucopy:0x618000263b80

堆栈信息

NSString内存小结,以及isEqual与isEqualToString的研究_第1张图片
屏幕快照 2017-06-08 下午7.37.40.png

上面我们看出str0、str2、strs0copy的内存地址是一样的

str1和str3copy是一样的

堆栈信息告诉我们str0、str2、strs0copy的类型是__NSCFConstantString;字符串常量存储区,字面量相等的共用一块常量存储地址,常量区的copy操作是浅拷贝依然相同地址

对于str2的写法,编译器会报一警告,警告我们str2的字面量赋值是多余的,默认会给我们str0的形式,str2依然是在常量区;

str1和str3copy的类型是NSTaggedPointerString,跟字面量的值密切相关,是在栈上

其余的str3、str0mucopy、str00mucopy则是__NSCFString,堆分配

关于NSTaggedPointerString的解释参考
【译】采用Tagged Pointer的字符串
深入理解Tagged Pointer

isEqualToString、isEqual

isEqual比较两个两个对象是否相等
来看下isEqual的源码

- (BOOL)isEqual:(id)obj {
    return obj == self;
}

回过头,我们看开头的输出结果分别是0、1、1,既然str0和str1的内存地址是不一样的,那为什么isEqual会相等呢,那就是NSString内部重写了isEqual,虽然拿不到官方的源码,我们可以看下GNUstep Base源码一定程度可以作为一个参考

- (BOOL) isEqual: (id)anObject
{
  if (anObject == self)
    {
      return YES;
    }
  if (anObject != nil && [anObject isKindOfClass: NSStringClass])
    {
      return [self isEqualToString: anObject];
    }
  return NO;
}

内存不等的情况下比较的说isEqualToString,这样我们就明白了
我们再看下isEqualToString

- (BOOL) isEqualToString: (NSString*)aString
{
  if (aString == self)
    {
      return YES;
    }
  if ([self hash] != [aString hash])
    {
      return NO;
    }
  if (strCompNsNs(self, aString, 0, (NSRange){0, [self length]})
    == NSOrderedSame)
    {
      return YES;
    }
  return NO;
}

首先是否相等比较,然后比较hash值,最后比较字面量是否相同。
至此回到开头我们荡然于胸了。
当然对于我们自定义对象的比较可以重写- (BOOL) isEqual:和- (NSUInteger)hash,当然这里就不展开balabala了

后记

以上也是随手看某一源码随手细致的琢磨了下,有不同见解的欢迎指出,生命在于折腾尤其在我大天朝=。=

你可能感兴趣的:(NSString内存小结,以及isEqual与isEqualToString的研究)