一尘不染

使用Swift将NSTask实时输出到NSTextView

swift

我正在使用NSTask运行rsync,并且希望状态显示在窗口内滚动视图的文本视图中。现在我有这个:

let pipe = NSPipe()
task2.standardOutput = pipe
task2.launch()

let data = pipe.fileHandleForReading.readDataToEndOfFile()
let output: String = NSString(data: data, encoding: NSASCIIStringEncoding)! as String

textView.string = output

这就是我有关传输的一些统计信息,但我想实时获取输出,例如在Xcode中运行该应用程序并将其放入文本视图时所打印的内容。有没有办法做到这一点?


阅读 430

收藏
2020-07-07

共1个答案

一尘不染

您可以使用通知从管道异步读取。这是一个演示其工作原理的简单示例,希望可以帮助您入门:

let task = NSTask()
task.launchPath = "/bin/sh"
task.arguments = ["-c", "echo 1 ; sleep 1 ; echo 2 ; sleep 1 ; echo 3 ; sleep 1 ; echo 4"]

let pipe = NSPipe()
task.standardOutput = pipe
let outHandle = pipe.fileHandleForReading
outHandle.waitForDataInBackgroundAndNotify()

var obs1 : NSObjectProtocol!
obs1 = NSNotificationCenter.defaultCenter().addObserverForName(NSFileHandleDataAvailableNotification,
    object: outHandle, queue: nil) {  notification -> Void in
        let data = outHandle.availableData
        if data.length > 0 {
            if let str = NSString(data: data, encoding: NSUTF8StringEncoding) {
                print("got output: \(str)")
            }
            outHandle.waitForDataInBackgroundAndNotify()
        } else {
            print("EOF on stdout from process")
            NSNotificationCenter.defaultCenter().removeObserver(obs1)
        }
}

var obs2 : NSObjectProtocol!
obs2 = NSNotificationCenter.defaultCenter().addObserverForName(NSTaskDidTerminateNotification,
    object: task, queue: nil) { notification -> Void in
        print("terminated")
        NSNotificationCenter.defaultCenter().removeObserver(obs2)
}

task.launch()

而是print("got output: \(str)")可以将接收到的字符串附加到文本视图中。

上面的代码假定运行循环处于活动状态(在默认的Cocoa应用程序中就是这种情况)。

2020-07-07