我在Swift中构建了一个OS X命令行工具(Objective-C中的相同问题),用于下载某些文件。我正在尝试使用下载进度更新命令行。不幸的是,我无法阻止print语句跳转到下一行。
根据我的研究,回车\r
应该跳到相同行的开头(而\n
会插入一个新行)。
所有测试都在OS X终端应用程序中执行,而不是在Xcode控制台上执行。
let logString = String(format: "%2i%% %.2fM \r", percentage, megaBytes)
print(logString)
仍然,显示器插入一个新行。如何防止这种情况?
答案 0 :(得分:6)
注意:这在Xcode的调试器窗口中不起作用,因为它不是真正的终端模拟器,并且不能完全支持转义序列。因此,为了测试,你必须编译它,然后在终端窗口中手动运行它。
\r
应该可以在大多数终端中移动到当前行的开头,但您还应该查看VT100 Terminal Control Escape Sequences。他们通过在Swift中发送转义字符\u{1B}
,然后发出命令来工作。 (警告:他们制作一些非常难看的字符串定义)
您可能需要的是\u{1B}[K
,它会清除从当前光标位置到结尾的行。如果您不这样做,并且您的进度输出长度不一样,那么您将从之前的打印语句中留下遗留物。
其他一些有用的是:
\u{1B}[\(y);\(x)H
注意:x
和y
是Int
插入字符串插值。 \u{1B}7
\u{1B}8
\u{1B}[2J
您还可以执行有趣的操作,例如设置文本前景色和背景色。
如果由于某种原因您无法使\r
工作,您可以在打印logString
之前保存光标状态/位置然后在以下情况之后恢复它来解决此问题:
let logString = String(format: "\u{1B}7%2i%% %.2fM \u{1B}8", percentage, megaBytes)
print(logString)
或者在打印之前移动到预定义的(x,y)位置:
let x = 0
let y = 1 // Rows typically start at 1 not 0, but it depends on the terminal and shell
let logString = String(format: "\u{1B}[\(y);\(x)H%2i%% %.2fM ", percentage, megaBytes)
print(logString)
答案 1 :(得分:4)
您的示例仅适用于实际的命令行,而不适用于调试器控制台。而且你还需要为每次迭代刷新stdout,如下所示:
var logString = String(format: "%2i%% %.2fM \r", 10, 5)
print(logString)
fflush(__stdoutp)
答案 2 :(得分:2)
在此示例中,假设一些异步任务正在通过传递Progress类型的回调进行。
// Print an empty string first otherwise whichever line is above the printed out progress will be removed
print("")
let someProgressCallback: ExampleAsyncCallbackType = { progress in
let percentage = Int(progress.fractionCompleted * 100)
print("\u{1B}[1A\u{1B}[KDownloaded: \(percentage)%")
}
关键部分是\u{1B}[1A\u{1B}[K
,然后是要打印到屏幕上的所有内容。