1

我正在开发一个 OS X 应用程序,该应用程序一直运行并处理他队列中的所有内容。这些作业将其进程记录在标准输出中。我正在尝试使用在后台线程中运行的 while 循环实时从标准输出中获取此信息。问题是运行 12 小时后,内存使用量从 25MB 上升到 >750MB。该应用程序正在使用 ARC,所以我在这里有点迷路。任何人都知道如何解决这个问题?

NSFileHandle *fileStd = [stdPipe fileHandleForReading];

NSTask *task = [[NSTask alloc] init];
[task setLaunchPath: cmd];
[task setArguments: arguments];

NSPipe *stdPipe = [NSPipe pipe];
NSPipe *errPipe = [NSPipe pipe];

[task setStandardOutput: stdPipe];
[task setStandardError: errPipe];

NSFileHandle *fileStd = [stdPipe fileHandleForReading];

[task launch];

NSData *readData;
NSString *logString;

while ((readData = [fileStd availableData]) && [readData length]){

   logString = [[NSString alloc] initWithData: readData encoding: NSUTF8StringEncoding];
        [self parseLog:logString];
    }

    readData = nil;
}

[task waitUntilExit];

和 parseLog 方法:

+ (void) parseLog:(NSString *)output {

    NSArray *chunks = [output componentsSeparatedByString: @"\r"];
    NSString *lastLine = [chunks lastObject];

    [lastLine writeToFile:@"/tmp/status.log" atomically:YES encoding:NSUTF8StringEncoding error:nil];

    NSRange textRange = [lastLine rangeOfString:@"<TEST> "];
    if(textRange.location != NSNotFound) {

        NSMutableDictionary *info = [NSMutableDictionary dictionary];
        [info setObject:lastLine forKey:@"test"];

        [[NSNotificationCenter defaultCenter] postNotificationName:@"handleNotification" object:nil userInfo:info];
    }
}
4

2 回答 2

4

使用分配工具查看堆增长的原因。

特别是,您可以使用 Heapshot 分析来非常有效地找出此类问题。

这里有一个写法。

简而言之:

  • 在分配工具下启动您的应用程序
  • (1)按下标记堆按钮
  • 等一会儿
  • 转到 (1)

重复一段时间。在您的情况下,您可能希望每 20 分钟到一个小时按一次标记堆按钮。


杰里米几乎做对了。你需要稍微重构你的循环:

while (1) {
@autoreleasepool {
   if (readData = [fileStd availableData]) && [readData length]){

   logString = [[NSString alloc] initWithData: readData encoding: NSUTF8StringEncoding];
        [self parseLog:logString];
    }

    readData = nil;
   } else {
    break;
   }
}

readData需要在游泳池里。

于 2013-07-16T15:51:25.000 回答
3

ARC 仍然在“幕后”使用引用计数。特别是,它仍然使用自动释放池,并且您不会在任何地方耗尽任何自动释放池。尝试用一个块封闭你的 while 循环@autoreleasepool体。

while ((readData = [fileStd availableData]) && [readData length]){
    @autoreleasepool
    {

       logString = [[NSString alloc] initWithData: readData encoding: NSUTF8StringEncoding];
       [self parseLog:logString];
    }
}

这相当于在循环开始时创建一个自动释放池并在结束时将其排空。

编辑

正如 bbum 所说, readData 在每次迭代时都放入一个自动释放池中,但它@autoreleasepool在位之外,因此代码需要重组:

bool dataToProcess = true;
while (dataToProcess){
    @autoreleasepool
    {
       NSData* readData = [fileStd availableData];
       dataToProcess = [readData length] > 0;
       if (dataToProcess)
       {
           logString = [[NSString alloc] initWithData: readData encoding: NSUTF8StringEncoding];
           [self parseLog:logString];
       }
    }
}
于 2013-07-16T16:03:14.280 回答