跑步时
top -n1 | head
终端的光标消失。跑步top -n1
会让它回来。
gnome-terminal
在tilix
Ubuntu 16.04 和 CentOS 7.5 中进行了测试。
运行top -n1 | tail
没有这个问题,所以我认为,top
输出末尾的某些内容让光标重新出现,这在打印唯一时不会执行head
。
是什么原因导致这种情况以及如何更优雅地恢复光标?
答案1
恕我直言,最好的方法是top
使用“批处理”模式(-b
标志),该模式旨在与非交互式用例一起使用,例如管道到另一个程序或文件。
所以这
top -n1 -b | head
不会在没有光标的情况下离开 shell。
至于为什么光标消失了...
由于top
它是一个交互式程序,它会“扰乱”终端以获取输入、滚动内容等,并且它隐藏光标。
当终止时,它必须恢复光标和被调用之前发现的显示状态,并且它通过发送一个或多个控制码到终端本身。
通过管道传输命令head
,该控制代码将无法通过(head
默认情况下仅打印前 10 行,并且两者的输出top
以及用于恢复终端状态的控制代码始终> 10 行)。
事实上,如果您提供head
足够的行数来打印,光标就会出现!
例如,
top -n1 | head -n 100
在我的系统上留下光标。
答案2
我无法在所有地方重现此行为,但它确实出现在 Ubuntu 18.04 上
检查顶部输出的十六进制转储是有启发性的:
$ top -n1 | head -n1 | xxd
00000000: 1b5b 3f31 681b 3d1b 5b3f 3235 6c1b 5b48 .[?1h.=.[?25l.[H
00000010: 1b5b 324a 1b28 421b 5b6d 746f 7020 2d20 .[2J.(B.[mtop -
00000020: 3133 3a34 333a 3034 2075 7020 3120 6d69 13:43:04 up 1 mi
00000030: 6e2c 2020 3120 7573 6572 2c20 206c 6f61 n, 1 user, loa
00000040: 6420 6176 6572 6167 653a 2030 2e38 312c d average: 0.81,
00000050: 2030 2e35 342c 2030 2e32 321b 2842 1b5b 0.54, 0.22.(B.[
00000060: 6d1b 5b33 393b 3439 6d1b 2842 1b5b 6d1b m.[39;49m.(B.[m.
00000070: 5b33 393b 3439 6d1b 5b4b 0a [39;49m.[K.
$ top -n1 | tail -n1 | xxd
00000000: 1b5b 3f31 326c 1b5b 3f32 3568 1b5b 4b .[?12l.[?25h.[K
$
特别是,开始的序列0x1b5b3f
是ANSI 转义序列,它们实际上是控制光标位置和文本颜色等内容的元数据。
特别是,在顶部输出的第一行的开头,有ESC [?25l
,在最后一行的末尾有ESC [?25h
。根据维基百科页面,这些是隐藏和显示光标的相应代码。
通过将top -n1
输出管道传输到head
,终端将在开始时接收 hide-cursor 命令,但在结束时不会接收 show-cursor 命令,因此光标将保持不可见,直到某些其他操作再次将其打开。
@MrShunz 建议使用该-b
选项top
是正确的。此选项禁用 top 输出中的所有 ANSI 转义序列,而仅输出纯 ASCII 可打印文本。执行top
with期间不会损害游标-b
:
$ top -b -n1 | head -n1 | xxd
00000000: 746f 7020 2d20 3133 3a35 393a 3236 2075 top - 13:59:26 u
00000010: 7020 3138 206d 696e 2c20 2031 2075 7365 p 18 min, 1 use
00000020: 722c 2020 6c6f 6164 2061 7665 7261 6765 r, load average
00000030: 3a20 302e 3134 2c20 302e 3036 2c20 302e : 0.14, 0.06, 0.
00000040: 3037 0a 07.
$