使用Swift将NSTask实时输出到NSTextView

我正在使用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中运行该应用程序并将其放入文本视图时所打印的内容。有没有办法做到这一点?


三国纷争
浏览 986回答 3
3回答

慕村9548890

您可以使用通知从管道异步读取。这是一个演示其工作原理的简单示例,希望可以帮助您入门: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 = pipelet outHandle = pipe.fileHandleForReadingoutHandle.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应用程序中就是这种情况)。

达令说

从macOS 10.7开始,还有一个readabilityHandler属性NSPipe,您可以在该属性上设置何时有新数据可用的回调: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 = pipelet outHandle = pipe.fileHandleForReadingoutHandle.readabilityHandler = { pipe in    if let line = String(data: pipe.availableData, encoding: NSUTF8StringEncoding) {        // Update your view with the new text here        print("New ouput: \(line)")    } else {        print("Error decoding data: \(pipe.availableData)")    }}task.launch()我很惊讶没有人提到这件事,因为它要简单得多。
打开App,查看更多内容
随时随地看视频慕课网APP