从进程'stdout读取时,NSTask需要刷新,而终端则不需要

时间:2012-11-13 03:53:54

标签: python objective-c stdout pipe nstask

我有一个简单的Python脚本,询问你的名字,然后把它吐出来:

def main():
    print('Enter your name: ')
    for line in sys.stdin:
        print 'You entered: ' + line

非常简单的东西!在OS X终端中运行时,它运行良好:

$ python nameTest.py 
Enter your name: 
Craig^D
You entered: Craig

但是,当尝试通过NSTask 运行此过程时,只有在Python脚本中添加了额外的flush()调用时才会出现stdout。

这就是我配置NSTask和管道的方法:

NSTask *_currentTask = [[NSTask alloc] init];
_currentTask.launchPath = @"/usr/bin/python";
_currentTask.arguments = [NSArray arrayWithObject:@"nameTest.py"];

NSPipe *pipe = [[NSPipe alloc] init];
_currentTask.standardOutput = pipe;
_currentTask.standardError = pipe;

dispatch_queue_t stdout_queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0);

__block dispatch_block_t checkBlock;

checkBlock = ^{
    NSData *readData = [[pipe fileHandleForReading] availableData];
    NSString *consoleOutput = [[NSString alloc] initWithData:readData encoding:NSUTF8StringEncoding];
    dispatch_sync(dispatch_get_main_queue(), ^{
        [self.consoleView appendString:consoleOutput];
    });
    if ([_currentTask isRunning]) {
        [NSThread sleepForTimeInterval:0.1];
        checkBlock();
    } else {
        dispatch_sync(dispatch_get_main_queue(), ^{
            NSData *readData = [[pipe fileHandleForReading] readDataToEndOfFile];
            NSString *consoleOutput = [[NSString alloc] initWithData:readData encoding:NSUTF8StringEncoding];
            [self.consoleView appendString:consoleOutput];
        });
    }
};

dispatch_async(stdout_queue, checkBlock);

[_currentTask launch];

但是当运行NSTask时,它就是这样(它最初是空白的,但在输入我的名字并按下CTRL + D后,它会立即完成所有操作):

Craig^DEnter your name: 
You entered: Craig

所以,我的问题是:如何在我的stdout中阅读NSTask,而无需在我的Python脚本中添加额外的flush()语句?为什么输入您的姓名:提示在以NSTask格式运行时不会立即显示?

1 个答案:

答案 0 :(得分:5)

当Python发现其标准输出是终端时,它会安排在脚本从sys.stdout读取时自动刷新sys.stdin。使用NSTask运行脚本时,脚本的标准输出是管道,而不是终端。

更新

有一个特定于Python的解决方案。您可以将-u标志传递给Python解释器(例如_currentTask.arguments = @[ @"-u", @"nameTest.py"];),它告诉Python不要缓冲标准输入,标准输出或标准错误。您还可以在流程环境中设置PYTHONUNBUFFERED=1以达到相同的效果。

ORIGINAL

适用于任何程序的更通用的解决方案使用所谓的“伪终端”(或者,历史上称为“伪电传”),我们将其简化为“pty”。 (事实上​​,这就是终端应用程序本身所做的。它是一种罕见的Mac,它有一个物理终端或电传打字机连接到串口!)

每个pty实际上是一对虚拟设备:从设备和主设备。您写入主站的字节,您可以从从站读取,反之亦然。因此,这些设备更像是套管(双向),而不是像管道(单向)。此外,pty还允许您设置终端I / O标志(或“termios”),以控制从属设备是否回显其输入,是否一次传递其输入一行或一次传递一个字符等等。

无论如何,您可以使用openpty功能轻松打开主/从对。这里有一个小类,您可以使用它来使NSTask对象使用从属端进行任务的标准输入和输出。

NSTask + PTY.h

@interface NSTask (PTY)

- (NSFileHandle *)masterSideOfPTYOrError:(NSError **)error;

@end

NSTask + PTY.m

#import "NSTask+PTY.h"
#import <util.h>

@implementation NSTask (PTY)

- (NSFileHandle *)masterSideOfPTYOrError:(NSError *__autoreleasing *)error {
    int fdMaster, fdSlave;
    int rc = openpty(&fdMaster, &fdSlave, NULL, NULL, NULL);
    if (rc != 0) {
        if (error) {
            *error = [NSError errorWithDomain:NSPOSIXErrorDomain code:errno userInfo:nil];
        }
        return NULL;
    }
    fcntl(fdMaster, F_SETFD, FD_CLOEXEC);
    fcntl(fdSlave, F_SETFD, FD_CLOEXEC);
    NSFileHandle *masterHandle = [[NSFileHandle alloc] initWithFileDescriptor:fdMaster closeOnDealloc:YES];
    NSFileHandle *slaveHandle = [[NSFileHandle alloc] initWithFileDescriptor:fdSlave closeOnDealloc:YES];
    self.standardInput = slaveHandle;
    self.standardOutput = slaveHandle;
    return masterHandle;
}

@end

你可以像这样使用它:

NSTask *_currentTask = [[NSTask alloc] init];
_currentTask.launchPath = @"/usr/bin/python";
_currentTask.arguments = @[[[NSBundle mainBundle] pathForResource:@"nameTest" ofType:@"py"]];

NSError *error;
NSFileHandle *masterHandle = [_currentTask masterSideOfPTYOrError:&error];
if (!masterHandle) {
    NSLog(@"error: could not set up PTY for task: %@", error);
    return;
}

然后,您可以从任务中读取并使用masterHandle写入任务。