内存管理:NSString的stringWithCString:编码:

假设我从某个函数中获取了一个C字符串:

char * mystring = SomeCFunction(...);

我拥有这个字符串(当我完成时,我负责释放它)。

如果,在Objective-C中,我使用以下命令创建NSString *

 NSString * mynsstring = [NSString stringWithCString:mystring encoding:NSUTF8StringEncoding]; 

我还有责任释放原来的C字符串吗?

我假设答案是肯定的,但我在文档中找不到明确的答案 – 而相反方法( cStringUsingEncoding )的function虽然很明显,但确实让我暂停,因为它为你处理了cString释放。

如果答案是肯定的,那么我还要负责确保在使用NSString *之前我没有释放c字符串,或者该函数是否为我复制字符串? 我问这个是因为stringWithCString的文档说:

返回包含给定C数组中的字节的字符串,根据给定的编码进行解释。

这仍然让我想知道它是否真的复制了字节,或者只是在内部指向它们(我基本上只是在做一个演员)。

它会复制字节。

如果你不想复制字节,可以使用- (id)initWithBytesNoCopy:(void *)bytes length:(NSUInteger)length encoding:(NSStringEncoding)encoding freeWhenDone:(BOOL)flag

如果你使用- (id)initWithBytes:(const void *)bytes length:(NSUInteger)length encoding:(NSStringEncoding)encoding ,或任何商品方法如stringWithCString:encoding:它本身调用这个低级方法,则复制字节。

API不知道你传递它的指针来自哪里,所以它不知道如何取得它的所有权。 它可以来自malloc(),但它也可以是内部指针或来自mmap(),在这种情况下free()将不起作用。 因此,API被迫复制数据。

您可以将C缓冲区的所有权传递给Cocoa API的唯一时间是API允许您告诉它需要释放。 例如,看一下-[NSData initWithBytesNoCopy:length:freeWhenDone:]