在 awk 系列的第八节,咱们介绍了一些强大的 awk 命令功能,它们是变量、数字表达式和赋值运算符。linux
本节咱们将学习更多的 awk 功能,即 awk 的特殊模式:BEGIN
和 END
。git
学习 awk 的模式 BEGIN 和 ENDgithub
随着咱们逐渐展开,并探索出更多构建复杂 awk 操做的方法,将会证实 awk 的这些特殊功能的是多么强大。正则表达式
开始前,先让咱们回顾一下 awk 系列的介绍,记得当咱们开始这个系列时,我就指出 awk 指令的通用语法是这样的:shell
# awk 'script' filenames
在上述语法中,awk 脚本拥有这样的形式:windows
/pattern/ { actions }
你一般会发现脚本中的模式(/pattern/
)是一个正则表达式,不过你也能够将模式使用特殊模式 BEGIN
和 END
。所以,咱们也能按照下面的形式编写一条 awk 命令:bash
awk ' BEGIN { actions } /pattern/ { actions } /pattern/ { actions } ………. END { actions } ' filenames
假如你在 awk 脚本中使用了特殊模式:BEGIN
和 END
,如下则是它们对应的含义:dom
BEGIN
模式:是指 awk 将在读取任何输入行以前当即执行 BEGIN
中指定的动做。END
模式:是指 awk 将在它正式退出前执行 END
中指定的动做。含有这些特殊模式的 awk 命令脚本的执行流程以下:ide
BEGIN
模式,则 BEGIN
中全部的动做都会在读取任何输入行以前执行。END
模式,那么将会执行相应的动做。当你使用特殊模式时,想要在 awk 操做中得到最好的结果,你应当记住上面的执行顺序。学习
为了便于理解,让咱们使用第八节的例子进行演示,那个例子是关于 Tecmint 拥有的域名列表,并保存在一个叫作 domains.txt 的文件中。
news.tecmint.com tecmint.com linuxsay.com windows.tecmint.com tecmint.com news.tecmint.com tecmint.com linuxsay.com tecmint.com news.tecmint.com tecmint.com linuxsay.com windows.tecmint.com tecmint.com
$ cat ~/domains.txt
查看文件内容
在这个例子中,咱们但愿统计出 domains.txt 文件中域名 tecmint.com
出现的次数。因此,咱们编写了一个简单的 shell 脚本帮助咱们完成任务,它使用了变量、数学表达式和赋值运算符的思想,脚本内容以下:
#!/bin/bash for file in $@; do if [ -f $file ] ; then ### 输出文件名 echo "File is: $file" ### 输出一个递增的数字记录包含 tecmint.com 的行数 awk '/^tecmint.com/ { counter+=1 ; printf "%s\n", counter ; }' $file else ### 若输入不是文件,则输出错误信息 echo "$file 不是一个文件,请指定一个文件。" >&2 && exit 1 fi done ### 成功执行后使用退出代码 0 终止脚本 exit 0
如今让咱们像下面这样在上述脚本的 awk 命令中应用这两个特殊模式:BEGIN
和 END
:
咱们应当把脚本:
awk '/^tecmint.com/ { counter+=1 ; printf "%s\n", counter ; }' $file
改为:
awk ' BEGIN { print "文件中出现 tecmint.com 的次数是:" ; } /^tecmint.com/ { counter+=1 ; } END { printf "%s\n", counter ; } ' $file
在修改了 awk 命令以后,如今完整的 shell 脚本就像下面这样:
#!/bin/bash for file in $@; do if [ -f $file ] ; then ### 输出文件名 echo "File is: $file" ### 输出文件中 tecmint.com 出现的总次数 awk ' BEGIN { print "文件中出现 tecmint.com 的次数是:" ; } /^tecmint.com/ { counter+=1 ; } END { printf "%s\n", counter ; } ' $file else ### 若输入不是文件,则输出错误信息 echo "$file 不是一个文件,请指定一个文件。" >&2 && exit 1 fi done ### 成功执行后使用退出代码 0 终止脚本 exit 0
awk 模式 BEGIN 和 END
当咱们运行上面的脚本时,它会首先输出 domains.txt 文件的位置,而后执行 awk 命令脚本,该命令脚本中的特殊模式 BEGIN
将会在从文件读取任何行以前帮助咱们输出这样的消息“文件中出现 tecmint.com 的次数是:
”。
接下来,咱们的模式 /^tecmint.com/
会在每一个输入行中进行比较,对应的动做 { counter+=1 ; }
会在每一个匹配成功的行上执行,它会统计出 tecmint.com
在文件中出现的次数。
最终,END
模式将会输出域名 tecmint.com
在文件中出现的总次数。
$ ./script.sh ~/domains.txt
用于统计字符串出现次数的脚本
最后总结一下,咱们在本节中演示了更多的 awk 功能,并学习了特殊模式 BEGIN
和 END
的概念。
正如我以前所言,这些 awk 功能将会帮助咱们构建出更复杂的文本过滤操做。第十节将会给出更多的 awk 功能,咱们将会学习 awk 内置变量的思想,因此,请继续保持关注。
via: http://www.tecmint.com/learn-use-awk-special-patterns-begin-and-end/
做者:Aaron Kili
译者:ChrisLeeGit
校对:wxy