echo -e 'one two three\nfour five six\nseven eight nine'
one two three
four five six
seven eight nine
我怎样才能做一些“魔术”来获得这个输出?:
three
six
nine
更新:我不需要这种特定的方式,我需要一个通用的解决方案,以便无论一行中有多少列,例如:awk 始终显示最后一列。
答案1
尝试:
echo -e 'one two three\nfour five six\nseven eight nine' | awk '{print $NF}'
答案2
这比你想象的要容易。
$ echo one two three | awk '{print $NF}'
three
答案3
尝试(更短/更简单,但由于正则表达式的使用grep
而慢 3 倍):awk
grep -o '\S\+$' <(echo -e '... seven eight nine')
或者ex
(甚至更慢,但它在完成后打印整个缓冲区,当需要就地排序或编辑时更有用):
ex -s +'%s/^.*\s//g' -c'%p|q!' <(echo -e '... seven eight nine')
ex +'%norm $Bd0' -sc'%p|q!' infile
要就地更改,请替换-sc'%p|q!'
为-scwq
。
或者bash
:
while read line; do arr=($line); echo ${arr[-1]}; done < someinput
表现
通过以下方式生成 1GB 文件:
$ hexdump -C /dev/urandom | rev | head -c1G | pv > datafile
我已经执行了解析时间统计(运行了约 3 倍,并取得了最低值,在 MBP OS X 上进行了测试):
使用
awk
:$ time awk '{print $NF}' datafile > /dev/null real 0m12.124s user 0m10.704s sys 0m0.709s
使用
grep
:$ time grep -o '\S\+$' datafile > /dev/null real 0m36.731s user 0m36.244s sys 0m0.401s $ time grep -o '\S*$' datafile > /dev/null real 0m40.865s user 0m39.756s sys 0m0.415s
使用
perl
:$ time perl -lane 'print $F[-1]' datafile > /dev/null real 0m48.292s user 0m47.601s sys 0m0.396s
使用
rev
+cut
:$ time (rev|cut -d' ' -f1|rev) < datafile > /dev/null $ time rev datafile | cut -d' ' -f1 | rev > /dev/null real 1m10.342s user 1m19.940s sys 0m1.263s
使用
ex
:$ time ex +'%norm $Bd0_' -sc'%p|q!' datafile > /dev/null real 3m47.332s user 3m42.037s sys 0m2.617s $ time ex +'%norm $Bd0' -sc'%p|q!' datafile > /dev/null real 4m1.527s user 3m44.219s sys 0m6.164s $ time ex +'%s/^.*\s//g' -sc'%p|q!' datafile > /dev/null real 4m16.717s user 4m5.334s sys 0m5.076s
使用
bash
:$ time while read line; do arr=($line); echo ${arr[-1]}; done < datafile > /dev/null real 9m42.807s user 8m12.553s sys 1m1.955s
答案4
... | perl -lane 'print $F[-1]'