Awk 命令的多種用法,你知道幾個(gè)?
awk 命令提供的不僅僅是從輸入字符串中選擇字段,還包括提取數(shù)據(jù)列、打印簡(jiǎn)單文本、評(píng)估內(nèi)容——甚至進(jìn)行數(shù)學(xué)運(yùn)算。
如果只在需要從文本行中選擇特定字段時(shí)才使用 awk,你可能會(huì)錯(cuò)過(guò)該命令可以提供的許多其他用處。在本文中,我們將看看這個(gè)簡(jiǎn)單的用法以及 awk 可以為你做的其他事情,并通過(guò)足夠的示例向您展示該命令比想象的要靈活得多。
提取數(shù)據(jù)列
awk 提供的最簡(jiǎn)單的用法是從文件或通過(guò)管道傳輸?shù)剿臄?shù)據(jù)中選擇特定字段。默認(rèn)使用空格作為字段分隔符:
[root@localhost ~]# echo one two three four five | awk '{print $4}'
four
[root@localhost ~]# who | awk '{print $1}'
root
root
在上面顯示的命令中,awk 僅從提供的數(shù)據(jù)中提取第四個(gè)字段和第一個(gè)字段。
awk 還可以通過(guò)在 awk 命令后添加文件名來(lái)從文件中提取文本。
[root@localhost ~]# awk -F ':' '{print "user: "$1,"home: "$6,"shell: "$NF}' /etc/passwd
在這種情況下,awk 使用-F ':'選項(xiàng)指定 冒號(hào) 作為分隔符。選出了文本中的第一列、第六列和最后一列。
命令中的 $NF? 選擇最后一個(gè)字段。這是因?yàn)?NF 表示一行中的字段數(shù),而$NF 表示最后一個(gè)字段的值。
字段可以按你認(rèn)為有用的任何順序打印。在此示例中,我們正在重新排列 date 命令輸出中的字段。
[root@localhost ~]# date | awk '{print "Date:",$6,$2,$3}'
Date: 2021 Sep 26
如果你希望輸出由其他符號(hào)來(lái)做分隔符,你可以使用 OFS(輸出字段分隔符)指定輸出分隔符,如下所示:
[root@localhost ~]# date | awk '{OFS="-"; print $6,$2,$3}'
2021-Sep-26
評(píng)估內(nèi)容
還可以使用 awk 評(píng)估字段。例如,如果只想在 /etc/passwd 文件中列出UID大于1000的用戶信息:
[root@localhost ~]# awk -F ':' '$3 >= 1000' /etc/passwd
nobody:x:65534:65534:Kernel Overflow User:/:/sbin/nologin
user01:x:1000:1000::/home/user01:/bin/bash
test01:x:1001:1001::/home/test01:/bin/bash
natasha:x:1002:1002::/home/natasha:/bin/bash
如果要為列表添加標(biāo)題,可以添加 BEGIN 子句,如果標(biāo)題想要多行,可以使用\n換行符:
[root@localhost ~]# awk -F ':' 'BEGIN {print "user accounts\n=============="} $3 >= 1000' /etc/passwd
user accounts
==============
nobody:x:65534:65534:Kernel Overflow User:/:/sbin/nologin
user01:x:1000:1000::/home/user01:/bin/bash
test01:x:1001:1001::/home/test01:/bin/bash
natasha:x:1002:1002::/home/natasha:/bin/bash
用awk做數(shù)學(xué)計(jì)算
awk 提供了算數(shù)能力,可以計(jì)算平方根、對(duì)數(shù)、切線等。
[root@localhost ~]# awk 'BEGIN {print sqrt(2021)}'
44.9555
[root@localhost ~]# awk 'BEGIN {print log(2019)}'
7.61036
編寫(xiě) awk 腳本
還可以使用 awk 編寫(xiě)?yīng)毩⒛_本。這是一個(gè)模仿前面提供的示例之一的示例,但也計(jì)算了系統(tǒng)上擁有帳戶的用戶數(shù)。
[root@localhost ~]# vim list_users
#!/usr/bin/awk -f
# This line is a comment
BEGIN {
printf "%s\n","User accounts:"
print "=============="
FS=":"
n=0
}
# Now we'll run through the data
{
if ($3 >= 1000) {
print $1
n ++
}
}
END {
print "=============="
print n " accounts"
}
請(qǐng)注意,僅在腳本啟動(dòng)時(shí)運(yùn)行的 BEGIN 部分,提供標(biāo)題、指示字段分隔符并設(shè)置從 0 開(kāi)始的計(jì)數(shù)器。該腳本還包括一個(gè) END 部分,該部分僅在腳本最后運(yùn)行。
[root@localhost ~]# ./list_users /etc/passwd
User accounts:
==============
nobody
systemd-coredump
systemd-resolve
polkitd
sssd
chrony
unbound
user01
test01
natasha
==============
10 accounts
計(jì)算文件中的行數(shù)
要使用 awk 打印文件中的行數(shù),請(qǐng)執(zhí)行以下操作:
[root@localhost ~]# awk 'END {print NR}' /etc/passwd
26
命令中包含 END 意味著在處理行之后提供輸出。NR(記錄數(shù))表示文件中的行數(shù)。
獲取最常使用的命令
還可以將 awk 與許多其他命令一起使用,以查看在當(dāng)前歷史文件里面你最常使用的命令。
[root@localhost ~]# history | awk '{print $2}'|sort|uniq -c|sort -nr|head -5
58 lvs
48 ll
31 vim
29 vgs
29 df
? ?