思考
一言不合先上个小小的思考吧
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
堆栈信息
上面我们看出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了
后记
以上也是随手看某一源码随手细致的琢磨了下,有不同见解的欢迎指出,生命在于折腾尤其在我大天朝=。=