1

我正在尝试使用 NSTask 运行 UNIX 'apropos' 命令。这是我的代码:

NSTask *apropos = [[NSTask alloc] init];
NSPipe *pipe = [[NSPipe alloc] init];
[apropos setLaunchPath:@"/usr/bin/apropos"];
[apropos setArguments:[NSArray arrayWithObjects:@"filename", @"match", nil]];
[apropos setStandardOutput:pipe];
[apropos launch];
[apropos waitUntilExit];

问题是这永远不会回来。我还尝试使用 Apple 的示例代码 (TaskWrapper),它返回输出(分三段),但它从不调用 processFinished 处理程序。

此外, appendOutput: 处理程序接收重复​​项。因此,例如,如果 apropos 返回以下内容:

1 2 3 4 5

我可能会收到这样的信息:

1 2 3

1 2 3 4

5

(分为 3 条附加消息)。

我注意到 Apropos 以一种可以在命令行中上下滚动的格式显示输出,而不是直接将数据直接输出到标准输出;如何通过 NSTask 和 NSPipe 可靠地阅读此内容?

4

2 回答 2

3

我刚刚测试了这个程序,它运行良好:程序终止并/tmp/apropos.txt包含apropos.

#import <Foundation/Foundation.h>

int main()
{
    NSAutoreleasePool *pool = [NSAutoreleasePool new];

    NSTask *apropos = [[[NSTask alloc] init] autorelease];
    NSPipe *pipe = [[[NSPipe alloc] init] autorelease];
    NSFileHandle *readHandle = [pipe fileHandleForReading];
    [apropos setLaunchPath:@"/usr/bin/apropos"];
    [apropos setArguments:[NSArray arrayWithObjects:@"filename", @"match", nil]];
    [apropos setStandardOutput:pipe];
    [apropos launch];
    [apropos waitUntilExit];

    NSString *output = [[[NSString alloc]
        initWithData:[readHandle readDataToEndOfFile]
            encoding:NSUTF8StringEncoding] autorelease];

    [output writeToFile:@"/tmp/apropos.txt" atomically:YES
        encoding:NSUTF8StringEncoding error:NULL];

    [pool drain];
    return 0;
}

您是否有机会NSLog()检查输出?如果是这样,您可能需要按照我对 NSTask 相关问题的回答中的stdin说明设置管道。似乎发送数据会影响.NSLog()stderrNSTask

于 2011-01-22T02:00:49.373 回答
2

使用您的原始代码,我想这是因为您没有阅读命令的输出。管道只有有限的缓冲区大小,如果您不读取任务的输出,它可能会挂起等待缓冲区清空。我对您尝试的示例代码一无所知,所以我无能为力。至于最后一个问题,apropos只有在连接到终端时才使用寻呼机。你不是在模拟终端,所以你不必担心。您可以通过apropos whatever | cat在终端中运行并验证没有调用寻呼机来证明这一点。

于 2011-01-22T01:35:00.273 回答