awk内置变量
FILENAME awk浏览的文件名 NR 已读的记录数 NF 浏览记录的域的个数
ARGC 命令行参数个数 ARGV 命令行参数排列 ENVIRON 支持队列中系统环境变量的使用
FNR 浏览文件的记录数 FS 设置输入域分隔符,等价于命令行 -F选项
OFS 输出域分隔符 ORS 输出记录分隔符 RS 控制记录分隔符
例子:
统计/etc/passwd:文件名,每行的行号,每行的列数,对应的完整行内容:
awk -F ':' '{print "filename:" FILENAME ",linenumber:" NR ",columns:" NF ",linecontent:"$0}' /etc/passwd
使用printf替代print,可以让代码更加简洁,易读
awk -F ':' '{printf("filename:%10s,linenumber:%s,columns:%s,linecontent:%s\n",FILENAME,NR,NF,$0)}' /etc/passwd
awk编程
变量和赋值
统计/etc/passwd的账户人数
awk '{count++;print $0;} END{print "user count is ", count}' /etc/passwd
count是自定义变量。之前的action{}里都是只有一个print,其实print只是一个语句,而action{}可以有多个语句,以;号隔开
上面没有初始化count,虽然默认是0,但是妥当的做法还是初始化为0:
awk 'BEGIN {count=0;print "[start]user count is ", count} {count=count+1;print $0;} END{print "[end]user count is ", count}' /etc/passwd
统计某个文件夹下的文件占用的字节数
ls -l |awk 'BEGIN {size=0;} {size=size+$5;} END{print "[end]size is ", size}'
以M为单位显示:
ls -l |awk 'BEGIN {size=0;} {size=size+$5;} END{print "[end]size is ", size/1024/1024,"M"}'
条件语句
统计某个文件夹下的文件占用的字节数,过滤4096大小的文件(一般都是文件夹):
ls -l |awk 'BEGIN {size=0;print "[start]size is ", size} {if($5!=4096){size=size+$5;}} END{print "[end]size is ", size/1024/1024,"M"}'
循环语句
数组 因为awk中数组的下标可以是数字和字母,数组的下标通常被称为关键字(key)。值和关键字都存储在内部的一张针对key/value应用hash的表格里。由于hash不是顺序存储,因此在显示数组内容时会发现,它们并不是按照你预料的顺序显示出来的。数组和变量一样,都是在使用时自动创建的,awk也同样会自动判断其存储的是数字还是字符串。一般而言,awk中的数组用来从记录中收集信息,可以用于计算总和、统计单词以及跟踪模板被匹配的次数等等。
显示/etc/passwd的账户
awk -F ':' 'BEGIN {count=0;} {name[count] = $1;count++;}; END{for (i = 0; i < NR; i++) print i, name[i]}' /etc/passwd
awk处理多个文件
awk的数据输入有两个来源,标准输入和文件,后一种方式支持多个文件
shell的Pathname Expansion方式:awk '{...}' *.txt
直接指定多个文件:awk '{...}' a.txt b.txt c.txt ...
处理 2 个文件
一种是 awk 'NR==FNR{...}NR>FNR{...}' file1 file2
# 读入file1的时候,已读入file1的记录数FNR一定等于awk已读入的总记录数NR,因为file1是awk读入的首个文件,故读入file1时执行前一个命令块{...} # 读入file2的时候,已读入的总记录数NR一定>读入file2的记录数FNR,故读入file2时执行后一个命令块{...}
另一种是 awk 'NR==FNR{...;next}{...}' file1 file2
# 读入file1时,满足NR==FNR,先执行前一个命令块,但因为其中有next命令,故后一个命令块{...}是不会执行的
awk中正则表达式
awk中的正则表达式用斜杠(/.../)包括起来
匹配:使用域号匹配表达式,使用符号'~'后紧跟正则表达式,也可以使用if语句
awk '{if($4~/good/)print $0}' readfile
awk '{$0 ~ /good/}' readfile
精确匹配:
awk '{if($3 ~ /字符串/) print $0}' readfile //包含字符串的所有记录都匹配,不精确
awk '$3 == "字符串" {print $0}' readfile //确保只有字符串得以匹配,精确匹配
不匹配 awk '{if($4 !~ /匹配字符串/) print $0}' readfile
小于 < 小于等于 <= 大于 >>
awk '{if($6<$7) print "xxx"}' readfile
设置大小写:为查询大小写信息,可以使用[ ]符号
awk '/[Gg]reen/' readfile //匹配green Green的行
任意字符