运行`top -n1 | 时光标消失头`

运行`top -n1 | 时光标消失头`

跑步时

top -n1 | head

终端的光标消失。跑步top -n1会让它回来。

gnome-terminaltilixUbuntu 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
$ 

特别是,开始的序列0x1b5b3fANSI 转义序列,它们实际上是控制光标位置和文本颜色等内容的元数据。

特别是,在顶部输出的第一行的开头,有ESC [?25l,在最后一行的末尾有ESC [?25h。根据维基百科页面,这些是隐藏和显示光标的相应代码。

通过将top -n1输出管道传输到head,终端将在开始时接收 hide-cursor 命令,但在结束时不会接收 show-cursor 命令,因此光标将保持不可见,直到某些其他操作再次将其打开。

@MrShunz 建议使用该-b选项top是正确的。此选项禁用 top 输出中的所有 ANSI 转义序列,而仅输出纯 ASCII 可打印文本。执行topwith期间不会损害游标-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.
$ 

相关内容