访问已发布的NSString不会导致应用程序崩溃

时间:2012-01-30 18:32:02

标签: objective-c ios xcode

我知道这可能与Stackoverflow上提出的问题完全相反,但今天我发生了一些非常奇怪的事情。 我试图向某人展示如何使用Instruments和NSZombies,所以我试图创建一个崩溃。

我宣布了一个NSString,发布了它,然后尝试访问它但应用程序没有崩溃

NSString* test = [[NSString alloc] init];
[test release];
NSlog(@"%@", test);

我甚至试图将它发布两次,但它仍然没有使应用程序崩溃,它只是打印出来。

任何人都可以向我解释一下我做错了什么或我逻辑中的缺陷在哪里?

谢谢


编辑:我也试过这样的东西但仍然没有崩溃

NSString* test = [[NSString alloc] init];
test = @"something";
NSlog(@"%@", test);
[test release];
NSlog(@"%@", test);

我甚至添加了两个连续版本,测试= nil;发布后,只是为了确保。

3 个答案:

答案 0 :(得分:9)

NSString有时可能表现得很奇怪。

在您的示例中,您分配了一个没有数据的NSString 这就是它不崩溃的原因。

在没有数据的情况下分配NSString毫无意义,因为NSString个对象是不可变的。
在这种情况下,NSString将返回一种singleton实例,这意味着您无法释放它(嗯,您可以,但它不起作用)。
每次以这种方式分配NSString对象时,都会返回相同的实例。

请改为尝试:

NSString * test = [ [ NSString alloc ] initWithCString: "hello, world" encoding: NSUTF8StringEncoding ];

[ test release ];

NSLog( @"%@", test );

然后它会像预期的那样崩溃......

要证明我之前解释过的内容,请尝试以下方法:

NSString * test1 = [ [ NSString alloc ] init ];
NSString * test2 = [ [ NSString alloc ] init ];

NSLog( @"OK: %p %p", test1, test2 );

您可以看到打印了相同的地址,这意味着只分配了一个对象(对于没有数据的NSString个对象,“单例”对象。)

修改

我在评论中看到你试图“给字符串一个值”
使用NSString是不可能的,因为它们是不可变的。

我猜你试过这个:

NSString * s = [ [ NSString alloc ] init ];

s = @"hello, world";

在这种情况下,您没有给出值

您正在重新指定指针!

这意味着您的s变量(它是一个指针)将指向另一个字符串对象 而且这也意味着你将无法访问以前分配的字符串,因为你只是丢失了它的指针。

如果您想要可以更改的字符串对象,请查看NSMutableString但请记住,更改指针值与更改对象的值不同!

答案 1 :(得分:2)

如果您查看分配了retainCountNSString的{​​{1}}上的@"some_string",那么您会看到它是[[NSString alloc] init]。这是因为编译器已经知道对象在编译时是什么,并且已经设法将其存储为文字。所以当你发布它时,它实际上并没有做任何事情。它不是在运行时分配的对象 - 它只是在您的应用程序二进制文件中。

你可以通过在Mac上做这样的事情来解决你的崩溃:

-1

现在,在编译时不再知道该字符串,因此它实际上是一个已分配的对象,并具有正保留计数等。

请注意,#import <Foundation/Foundation.h> int main(int argc, char** argv) { NSString *test = [[NSString alloc] initWithCString:argv[0] encoding:NSASCIIStringEncoding]; NSLog(@"test = %p", test); [test release]; NSLog(@"test = %p", test); [test release]; NSLog(@"test = %p", test); return 0; } (在Mac OS X Lion上)如果编号在编译时已知,也会出现此行为。这更有趣,因为它是标记指针的一个例子。请查看this blog article以进行讨论。这里有一些代码来展示它:

NSNumber

请注意,它也不会崩溃,只会打印#import <Foundation/Foundation.h> int main(int argc, char** argv) { NSNumber *test = [[NSNumber alloc] initWithInt:1]; NSLog(@"test = %p", test); [test release]; NSLog(@"test = %p", test); [test release]; NSLog(@"test = %p", test); return 0; } 3次。

答案 2 :(得分:2)

您的示例中有两件事情发生。

首先,你的发行版和你的NSLog之间没有其他任何东西会修改你的字符串存储的内存,所以它恰好仍然完好无损。你不能依赖它。

其次,系统有[[NSString alloc] init]的特例。它返回一个永远不会释放的__NSCFConstantString类型的预定义对象。如果打印其引用计数,您将看到它是2147483647(0x7fffffff)。这是一个神奇的数字,表示任何保留或释放对象的尝试都会被忽略。