これはサンプルテキストです。 (名前は20210622_090009です)
nvmeSerial Endpoint nvmeSpeed nvmeWidth
================================================================================
nvme0n1 c7:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme1n1 c8:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme2n1 c9:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme3n1 ca:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme4n1 85:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme5n1 86:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme6n1 87:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme7n1 88:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme8n1 41:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme9n1 42:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme10n1 43:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme11n1 44:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme12n1 45:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme13n1 46:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme14n1 47:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme15n1 48:00.0 Width x2 (downgraded)
nvme16n1 01:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme17n1 02:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme18n1 03:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme19n1 04:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme20n1 05:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme21n1 06:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme22n1 07:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme23n1 08:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme24n1 09:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme25n1 0a:00.0 Speed 32GT/s (ok) Width x2 (downgraded)
スクリプトは次のとおりです。
#! /bin/bash
IFS_old="$IFS"
IFS=$'\n'
for line in $(cat 20210622_090009.txt | tail -n 26 | cut -f 5 | awk '{print $2}' )
do
echo "$line"
done
IFS="$IFS_old"
exit 0
スクリプトの出力は次のとおりです。
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
32GT/s
速度に数値があるかどうかにかかわらず、nvmeSpeed(Ex:8GT / s)を取得したいと思います。
ご覧のとおり、nvmeSpeedinはnvme15n1
空白です。
出力は表示されません。
私の質問は次のとおりです
awkスペースをforループ入力に置き換えるには?
答え1
awk
一人でもできます。シェルスクリプトラッパーは必要なく、そのようなバロック様のものも確かに必要ありませんcat 20210622_090009.txt | tail -n 26 | cut -f 5 | awk '{print $2}'
。そして、どこでもシェルの読み込み中にループを使用しないでください(またはawkやPerlループなどの言語の出力に対してforを実行することを避ける必要があります)。シェルループを使用してテキストを処理するのはなぜ悪い習慣と見なされますか?理由)。
経験則:「awkの出力を繰り返したい」と思う場合は、「これを行うにはほぼ確実にawkを使用する必要があります」に変更するか、awkシェルラッパーの入出力リダイレクトを設定する必要があります。バッチ処理ジョブ。 Perlと他のほとんどの言語でも同様です。他の言語はシェルよりも優れた操作をし、シェルを使用しようとすると作業が難しくなります。
とにかく正確に8つの列()がある場合、NF == 8
次のスクリプトは列4を印刷します。列が8個未満の場合(NF < 8
)、空白行が印刷されます。どちらの場合も、各入力ファイルの先頭にある2つのヘッダー行を無視します(1つ以上のファイル名引数を処理できます。awk FNR < 3 {next}
では、NRは読み取られた行の総数、FNRは行です)現在の文書)。
$ awk 'FNR < 3 {next}; NF == 8 {print $4}; NF < 8 {print ""}' 20210622_090009.txt
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
32GT/s
答え2
固定幅フィールドがあるようですので、FIELDWIDTHSとgensub()にGNU awkを使用してください。
$ awk -v FIELDWIDTHS='16 12 24 *' '
NR>2 {
gsub(/^ *| *$/,"",$3)
print gensub(/.* ([^ ]+) .*/,"\\1",1,$3)
}
' file
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
8GT/s
32GT/s
上記の内容は、まず各フィールドの内容を幅で識別します。
$ cat file
nvmeSerial Endpoint nvmeSpeed nvmeWidth
================================================================================
nvme0n1 c7:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme1n1 c8:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
nvme15n1 48:00.0 Width x2 (downgraded)
nvme25n1 0a:00.0 Speed 32GT/s (ok) Width x2 (downgraded)
$ cat tst.awk
BEGIN { FIELDWIDTHS="16 12 24 *" }
NR != 2 {
print
for (i=1; i<=NF; i++) {
gsub(/^ *| *$/,"",$i)
print "\t" i, "<" $i ">"
}
print "-----"
}
$ awk -f tst.awk file
nvmeSerial Endpoint nvmeSpeed nvmeWidth
1 <nvmeSerial>
2 <Endpoint>
3 <nvmeSpeed>
4 <nvmeWidth>
-----
nvme0n1 c7:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
1 <nvme0n1>
2 <c7:00.0>
3 <Speed 8GT/s (ok)>
4 <Width x2 (downgraded)>
-----
nvme1n1 c8:00.0 Speed 8GT/s (ok) Width x2 (downgraded)
1 <nvme1n1>
2 <c8:00.0>
3 <Speed 8GT/s (ok)>
4 <Width x2 (downgraded)>
-----
nvme15n1 48:00.0 Width x2 (downgraded)
1 <nvme15n1>
2 <48:00.0>
3 <>
4 <Width x2 (downgraded)>
-----
nvme25n1 0a:00.0 Speed 32GT/s (ok) Width x2 (downgraded)
1 <nvme25n1>
2 <0a:00.0>
3 <Speed 32GT/s (ok)>
4 <Width x2 (downgraded)>
-----
その後、印刷する3番目のフィールド部分を選択するのは簡単です。たとえば、私のgensub()
ように使用します。上記の内容は、行にないフィールド、フィールド内の単語数などに関係なく機能します。
答え3
別のアプローチは固定幅フィールドを想定しています(フィールドの1つだけを抽出し、空でない場合は常に6文字の「Speed」で始まると仮定します)。
cut -c35-52 file | sed '1,2d;s/ .*//'
または「速度」を一致させます。
awk -F ' Speed +' 'NR>2 {sub(/ .*/,"",$2); print $2}' file
sed -E '1,2d;s/.* Speed +([^ ]+).*/\1/;t;c\\' file
perl -nE 'say m{\sSpeed\s+(\S+)} if $.>2' file
答え4
awk 'NR>2{if($4 ~ /^[0-9].*GT/){print $1" =======>" $4}else{if($4 !~ /^[0-9].*GT/){print $1"==================== doesnt contain speed==========================="}}}' filename
出力
nvme0n1 =======>8GT/s
nvme1n1 =======>8GT/s
nvme2n1 =======>8GT/s
nvme3n1 =======>8GT/s
nvme4n1 =======>8GT/s
nvme5n1 =======>8GT/s
nvme6n1 =======>8GT/s
nvme7n1 =======>8GT/s
nvme8n1 =======>8GT/s
nvme9n1 =======>8GT/s
nvme10n1 =======>8GT/s
nvme11n1 =======>8GT/s
nvme12n1 =======>8GT/s
nvme13n1 =======>8GT/s
nvme14n1 =======>8GT/s
nvme15n1==================== doesnt contain speed===========================
nvme16n1 =======>8GT/s
nvme17n1 =======>8GT/s
nvme18n1 =======>8GT/s
nvme19n1 =======>8GT/s
nvme20n1 =======>8GT/s
nvme21n1 =======>8GT/s
nvme22n1 =======>8GT/s
nvme23n1 =======>8GT/s
nvme24n1 =======>8GT/s
nvme25n1 =======>32GT/s