为什么启用ARC会发生此自动释放错误?

时间:2012-11-29 00:37:19

标签: objective-c automatic-ref-counting autorelease

我写了一个小的CLI程序来删除特定的Safari cookie。从功能上来说它很好,但是它会抛出关于“自动释放而没有游泳池”的对象的警告。我的项目启用了ARC,因此我没有任何自动释放池。

这是我的代码:

// NSLog replacement from http://stackoverflow.com/a/3487392/1376063
void IFPrint (NSString *format, ...) {
    va_list args;
    va_start(args, format);

    fputs([[[NSString alloc] initWithFormat:format arguments:args] UTF8String], stdout);
    fputs("\n", stdout);

    va_end(args);
}

int main(int argc, const char * argv[])
{
    NSString *urlSearchString;
    if (argc > 1) {
        urlSearchString = [[NSString alloc] initWithUTF8String:argv[1]];
    }
    else {
        IFPrint(@"No URL provided, quitting.");
        return 1;
    }

    NSHTTPCookieStorage *cookieStorage = [NSHTTPCookieStorage sharedHTTPCookieStorage];

    NSString *filterString = [[NSString alloc] initWithFormat:@"domain ENDSWITH '%@'", urlSearchString];
    NSPredicate *filter = [NSPredicate predicateWithFormat:filterString];

    NSArray *matchedCookies = [cookieStorage.cookies filteredArrayUsingPredicate:filter];

    for (int i = 0; i < matchedCookies.count; i++) {
        [cookieStorage deleteCookie:[matchedCookies objectAtIndex:i]];
    }

    IFPrint(@"Removed %li cookies", matchedCookies.count);
    return 0;

}

我得到的信息是:

objc[15502]: Object 0x107b2bf00 of class NSThread autoreleased with no pool in place - just leaking - break on objc_autoreleaseNoPool() to debug

哪个出现在Xcode调试器中或直接运行发布二进制文件时(略微偏离:不应该将这些消息从“发布”版本中删除?)。导致它的线似乎是:

NSHTTPCookieStorage *cookieStorage = [NSHTTPCookieStorage sharedHTTPCookieStorage];

同样,如果我在不传递参数的情况下运行它,我会收到类似的消息:

objc[15630]: Object 0x100114ed0 of class __NSCFString autoreleased with no pool in place - just leaking - break on objc_autoreleaseNoPool() to debug
objc[15630]: Object 0x100114f80 of class __NSCFData autoreleased with no pool in place - just leaking - break on objc_autoreleaseNoPool() to debug

这似乎来自我正在使用的IFPrint函数(但是当我提供正确的参数时,当我使用IFPrint时,这不会出现。)

我有点超出我的深度,有人能告诉我哪里(以及如何)我出错了?

3 个答案:

答案 0 :(得分:3)

ARC仍需要自动释放池。像[NSPredicate predicateWithFormat:filterString]这样的方法继续释放一个自动释放的对象(尽管你不再需要关心自己,因为ARC处理它)。此外,您调用的任何库方法的内部实现可能会在运行时创建任意多个自动释放的对象。

您应该通过@autoreleasepool机制将代码包装在自动释放池中。

答案 1 :(得分:1)

main的整个身体与@autoreleasepool一起包裹起来:

int main(int argc, const char * argv[])
{
  @autoreleasepool 
  {
    // your code
  }
}

答案 2 :(得分:1)

您需要做的只是在主要内容中添加自动释放池。

int main(int argc, const char * argv[])
{ 
     @autoreleasepool
     {
         //Your code
     }
}