为了账号安全,请及时绑定邮箱和手机立即绑定

使用Swift将NSTask实时输出到NSTextView

使用Swift将NSTask实时输出到NSTextView

三国纷争 2019-12-04 15:20:05
我正在使用NSTask运行rsync,并且希望状态显示在窗口内滚动视图的文本视图中。现在我有这个:let pipe = NSPipe()task2.standardOutput = pipetask2.launch()let data = pipe.fileHandleForReading.readDataToEndOfFile()let output: String = NSString(data: data, encoding: NSASCIIStringEncoding)! as StringtextView.string = output这就是我有关传输的一些统计信息,但是我想实时获取输出,例如在Xcode中运行该应用程序并将其放入文本视图时所打印的内容。有没有办法做到这一点?
查看完整描述

3 回答

?
慕村9548890

TA贡献1884条经验 获得超4个赞

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


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应用程序中就是这种情况)。


查看完整回答
反对 回复 2019-12-04
?
达令说

TA贡献1821条经验 获得超6个赞

从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 = pipe

let outHandle = pipe.fileHandleForReading


outHandle.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()

我很惊讶没有人提到这件事,因为它要简单得多。


查看完整回答
反对 回复 2019-12-04
  • 3 回答
  • 0 关注
  • 975 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信