我知道这可能与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;发布后,只是为了确保。
答案 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)
如果您查看分配了retainCount
或NSString
的{{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)。这是一个神奇的数字,表示任何保留或释放对象的尝试都会被忽略。