【问题标题】:NSTask requires flush when reading from a process' stdout, Terminal does not从进程的标准输出读取时,NSTask 需要刷新,终端不需要
【发布时间】:2012-11-13 03:53:53
【问题描述】:

我有一个简单的 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() 调用时才会出现标准输出。

这就是我配置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

所以,我的问题是:如何在我的 Python 脚本中不需要额外的 flush() 语句的情况下从我的 NSTask 中读取 stdout?为什么以NSTask 运行时,输入您的姓名: 提示没有立即出现?

【问题讨论】:

  • 您有问题吗?
  • @kindall 抱歉,如果 OP 不清楚 - 在底部添加了具体问题。
  • 我希望每个为 Python 脚本没有输出中间结果而苦苦挣扎的人,找到这个。你又帮我省了 5 个小时的麻烦!谢谢!

标签: python objective-c stdout pipe nstask


【解决方案1】:

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

更新

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

原创

适用于任何程序的更通用的解决方案使用所谓的“伪终端”(或历史上的“伪电传打字机”),我们将其简称为“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 读取任务并写入任务。

【讨论】:

  • 感谢 rob,一个绝妙的答案,几乎可以完美运行。一个问题是我似乎无法关闭 category 方法返回的NSFileHandle,它只是挂起。 (调用closeFile 是我模拟CTRL+D 的方式)另外,由于输入/输出发生在同一个句柄上,所以我写入标准输入的每个字符都会回显到标准输出。
  • 对于回显问题:尝试使用tcgetattr获取从设备的termios。清除c_lflag 字段中的ECHO 位,并将修改后的termios 放回带有tcsetattr 的从站。您可能需要阅读 tcgetattrtermios 手册页。
  • 对于挂起问题:我不知道发生了什么。您可能需要找到其他一些 termios 位进行调整。
猜你喜欢
  • 2018-07-20
  • 2011-09-07
  • 2013-06-29
  • 1970-01-01
  • 1970-01-01
  • 2021-11-01
  • 2023-03-23
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多