web-dev-qa-db-ja.com

`top -n1 | |を実行するとカーソルが消える頭 `

実行中

top -n1 | head

端末のカーソルが消えます。 top -n1を実行すると元に戻ります。

Ubuntu 16.04およびCentOS 7.5のgnome-terminalおよびtilixでテストされています。


top -n1 | tailを実行してもこの問題は発生しないため、topの出力の最後にカーソルが再表示され、headのみを印刷する場合は実行されません。

これは何が原因で、どのようにしてカーソルをよりエレガントに戻すことができますか?

11
pLumo

どこでもこの動作を再現することはできませんでしたが、Ubuntu 18.04では表示されます


上位の出力の16進ダンプを調べることは有益です。

$ 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コマンドを受け取りません。そのため、他のアクションによって変更されるまで、カーソルは非表示のままになります。再び。

-bオプションをtopに使用するという@MrShunzの提案は正しいです。このオプションは、topの出力ですべてのANSIエスケープシーケンスを無効にしますが、代わりにプレーンなASCII印刷可能なテキストを出力します。top-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.
$ 
5
Digital Trauma

IMHOの最良の方法は、topで「バッチ」モードを使用することです(-bフラグ)。これは、別のプログラムまたはファイルへのパイピングなどの非インタラクティブな使用例で使用することを目的としています。

したがって、この

top -n1 -b | head

カーソルなしでシェルを離れることはありません。

カーソルが消える理由 ...

topは対話型プログラムであるため、入力を取得したり、コンテンツをスクロールしたりするために、ターミナルで「混乱」しますhidesカーソル。

終了するときは、呼び出される前にカーソルと表示ステータスを復元する必要があり、1つ以上の制御コードを端末自体に送信することで復元します。

headを介してコマンドをパイプすることにより、この制御コードは通過しません(headはデフォルトで最初の10行だけを印刷し、topと制御の両方の出力を最終状態を復元するコードは常に> 10行です)。

実際、印刷するのに十分な行をhead与えると、カーソルが表示されます!

例えば、

top -n1 | head -n 100

私のシステムにカーソルを置きます。

17
Mr Shunz