10

可能重复:
为什么弱 NSString 属性没有在 iOS 中发布?

我是 Objective C 的新手,我有一些我自己无法回答的问题。我有一段代码用于测试 __weak 变量(当然,我使用的是 ARC):

NSString *myString = [[NSString alloc] initWithFormat:@"John"];
NSString * __weak weakString = myString;
myString = nil; //<-- release the NSString object
NSLog(@"string: %@", weakString);

上述代码的输出与预期一致,因为 weakString 是一个弱变量:

2013-01-02 11:42:27.481 ConsoleApp[836:303] string: (null)

但是当我将代码修改为:

NSString *myString = [[NSString alloc] initWithFormat:@"John"];
NSString * __weak weakString = myString;
NSLog(@"Before: %@", weakString); //<--- output to see if the __weak variable really works.
myString = nil;
NSLog(@"After: %@", weakString);

输出完全不是我所期望的:

2013-01-02 11:46:03.790 ConsoleApp[863:303] Before: John
2013-01-02 11:46:03.792 ConsoleApp[863:303] After: John

后一个 NSLog 的输出必须是 (nil) 而不是“John”。我试图在许多文档中搜索,但我没有找到这个案例的答案。有人能给出合理的解释吗?提前致谢。

4

2 回答 2

7

NSLog函数将传递的 NSString 保留在自动释放池中。因此,在 autorelease 池耗尽之前,zeroing-weak 变量不会被归零。例如:

__weak NSString* weakString = nil;

@autoreleasepool {
    NSString* myString = [[NSString alloc] initWithFormat:@"Foo"]; // Retain count 1
    weakString = myString;         // Retain count 1
    NSLog(@"A: %@", weakString);   // Retain count 2
    NSLog(@"B: %@", weakString);   // Retain count 3
    myString = nil;                // Retain count 2
    NSLog(@"C: %@", weakString);   // Retain count 3

    NSAssert(weakString != nil, @"weakString is kept alive by the autorelease pool");
} 

// retain count 0
NSAssert(weakString == nil, @"Autorelease pool has drained.");

为什么 NSLog 将字符串放入自动释放池?这是一个实现细节。

您可以使用调试器或 Instruments 来跟踪 NSString 实例的保留计数。确切的保留计数并不重要,但它确实揭示了幕后发生的事情。重要的是 NSString 实例在自动释放池耗尽时被释放。

于 2013-01-02T07:12:23.410 回答
0

我认为这只是一些实现细节。你的弱变量正在被清除,但不仅仅是立即清除。例如,这按预期工作:

NSString *myString = [[NSString alloc] initWithFormat:@"John"];
NSString * __weak weakString = myString;
@autoreleasepool {
    NSLog(@"Before: %@", weakString);
    myString = nil;
}
NSLog(@"After: %@", weakString); // nil
于 2013-01-02T07:12:53.317 回答