Objective-C:弱的attritube不能按预期工作

hoa*_*Cap 10 weak-references ios strong-references

可能重复:
为什么iOS中没有释放弱NSString属性?

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

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

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

2013-01-02 11:42:27.481 ConsoleApp[836:303] string: (null)
Run Code Online (Sandbox Code Playgroud)

但是当我将代码修改为:

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);
Run Code Online (Sandbox Code Playgroud)

输出完全不是我的预期:

2013-01-02 11:46:03.790 ConsoleApp[863:303] Before: John
2013-01-02 11:46:03.792 ConsoleApp[863:303] After: John
Run Code Online (Sandbox Code Playgroud)

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

Dar*_*ren 7

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

__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.");
Run Code Online (Sandbox Code Playgroud)

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

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