2

あなたの名前を尋ねて、それを吐き出す簡単な 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ただし、このプロセスを 経由で実行しようとすると、追加の flush() 呼び出しが Python スクリプトに追加された場合にのみ 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

だから、私の質問は次のとおりです。Python スクリプトに追加の flush() ステートメントを必要とせずに、どのようにstdoutmy からを読み取ることができますか? として実行すると、 Enter your name:プロンプトがすぐに表示されないNSTaskのはなぜですか?NSTask

4

1 に答える 1

5

Python は、標準出力が端末であることを認識するとsys.stdout、スクリプトが から読み取るときに自動的にフラッシュするように調整しますsys.stdin。を使用してスクリプトを実行するNSTaskと、スクリプトの標準出力は端末ではなくパイプになります。

アップデート

これに対する Python 固有の解決策があります。-uフラグを Python インタープリター (例: ) に渡すことができます_currentTask.arguments = @[ @"-u", @"nameTest.py"];。これにより、標準入力、標準出力、または標準エラーをまったくバッファーしないように Python に指示されます。PYTHONUNBUFFERED=1プロセスの環境に設定して、同じ効果を達成することもできます。

オリジナル

あらゆるプログラムに適用されるより一般的な解決策は、「疑似端末」(または、歴史的には「疑似テレタイプ」) と呼ばれるものを使用します。これは単に「pty」に短縮されます。(実際、これは Terminal アプリ自体が行うことです。シリアル ポートに接続された物理端末またはテレタイプを備えたまれな Mac です!)

各 pty は、実際には仮想デバイスのペア (スレーブ デバイスとマスター デバイス) です。マスターに書き込むバイトは、スレーブから読み取ることができ、その逆も可能です。したがって、これらのデバイスは、パイプ (一方向) よりもソケット (双方向) に似ています。さらに、pty を使用すると、スレーブがその入力をエコーするかどうか、一度に 1 行または 1 文字ずつ入力に渡すかどうかなどを制御する端末 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

于 2012-11-13T05:07:57.720 に答える