最近被 NSString 无法释放这件事搞得糊涂。
首先说一下,发现问题的过程:
有两个 weak properties 。
@property (nonatomic, weak) id a;
@property (nonatomic, weak) id sa;
按道理来说, 当一个对象初始化就赋给了一个 weak 对象, 这个对象就会立刻被释放掉, 但有一个特殊的情况, 看下面的代码和输出。
{
NSString *temp = @"sa";
NSMutableString *sa = [[NSMutableString alloc] initWithString:temp];
NSMutableArray *array = [NSMutableArray arrayWithObject:@"aaa"];
self.a = [array copy];
self.sa = [sa copy];
NSLog(@"array:%p", array);
NSLog(@"self.a:%p", self.a);
NSLog(@"self.sa:%p", self.sa);
NSLog(@"temp:%p", temp);
}
array:0x600000241ad0
self.a:0x0
self.sa:0xa000000000061732
temp:0x1081d0030
为什么 self.sa 没有被释放?
到这里我们来引入 Tagged Pointer
这个概念.
对于那些所需内存小于60位的字符串,它可以创建一个Tagged Pointer。其余的则被放置在真正的NSString对象里。这使得常用的短字符串的性能得到明显的提升。
NSNumber、NSDate等, 都是使用 Tagged Pointer
.
当你重复运行的时候, 发现 self.sa 的地址, 始终没有发生变化, 其实那并不是对象的地址, 而是直接指向数据的指针.
可以发现实际的地址 self.sa:0xa000000000061732
对应的数据格式如下:
0x2 - Length (2)
0x73 - 's'
0x61 - 'a'
当我改代码如下:
{
NSString *temp = @"saaaaaaaaaaa";
NSMutableString *sa = [[NSMutableString alloc] initWithString:temp];
NSMutableArray *array = [NSMutableArray arrayWithObject:@"aaa"];
self.a = [array copy];
self.sa = [sa copy];
NSLog(@"array:%p", array);
NSLog(@"self.a:%p", self.a);
NSLog(@"self.sa:%p", self.sa);
NSLog(@"temp:%p", temp);
}
array:0x600000055300
self.a:0x0
self.sa:0x0
temp:0x109c74030
发现 self.sa 也变为 nil 了, 因为当长度大于60的时候, Tagged Pointer
失效, 改用对象存储, 则初始化后立刻被释放掉了.
感谢 @sven 在 stackoverflow 的解答.