Linux文本处理三剑客之GNU awk的使用

awk: Aho, Weinberger, Kernighan,报告生成器,格式化文本输出

有多种版本:New awk(nawk),GNU awk(gawk)

gawk –模式扫描和处理语言

基本用法:

    awk[options] ‘program’ var=value file…

    awk[options] -f programfile var=value file…

    awk[options] 'BEGIN{ action;… } pattern{ action;… } END{ action;… }' file …

awk程序通常由:BEGIN语句块、能够使用模式匹配的通用语句块、END语句块,共3部分组成program通常是被单引号或双引号中

选项:

    -F 指明输入时用到的字段分隔符

    -v var=value: 自定义变量

awk语言

基本格式:awk [options] 'program' file…

program:pattern{action statements;..}

pattern和action:

    pattern部分决定动作语句何时触发及触发事件

    (BEGIN,END)

action statements对数据进行处理,放在{}内指明

    (print, printf)

分割符、域和记录

    awk执行时,由分隔符分隔的字段(域)标记$1,$2..$n称为域标识。$0为所有域,注意:和shell中变量$符含义不同

    文件的每一行称为记录

    省略action,则默认执行 print $0 的操作

awk工作原理

    第一步:执行BEGIN{action;… }语句块中的语句

    第二步:从文件或标准输入(stdin)读取一行,然后执行pattern{ action;… }语句块,它逐行扫描文件,从第一行到最后一行重复这个过程,直到文件全部被读取完毕。

    第三步:当读至输入流末尾时,执行END{action;…}语句块

    BEGIN语句块在awk开始从输入流中读取行之前被执行,这是一个可选的语句块,比如变量初始化、打印输出表格的表头等语句通常可以写在BEGIN语句块中

    END语句块在awk从输入流中读取完所有的行之后即被执行,比如打印所有行的分析结果这类信息汇总都是在END语句块中完成,它也是一个可选语句块

    pattern语句块中的通用命令是最重要的部分,也是可选的。如果没有提供pattern语句块,则默认执行{ print },即打印每一个读取到的行,awk读取的每一行都会执行该语句块

    

print格式:print item1, item2, …

要点:

    (1) 逗号分隔符

    (2) 输出的各item可以字符串,也可以是数值;当前记录的字段、变量或awk的表达式

    (3) 如省略item,相当于print $0

示例:

awk '{print "hello,awk"}'

所有标准输入之后都转变成标准输出的hello,awk

wKiom1fkh2mQBQWiAAAmFX-zunU651.png

awk –F: '{print}' /etc/passwd

打印/etc/passwd所有内容

wKiom1fkiVXyRj0eAADI8QA51nM235.png

awk –F: ‘{print “root”}’ /etc/passwd

以冒号为分隔符,替换成root

wKiom1fkiF6Cr8SCAAAmbavZgJo309.png

awk –F: ‘{print $1}’ /etc/passwd

以冒号为分隔符;打印/etc/passwd文件的第一个参数

wKiom1fkiH3hFAbLAAApXSTfXDE837.png

awk –F: ‘{print $0}’ /etc/passwd

$0相当于正行,这里相当于打印所有

wKiom1fkihexMDAWAACmIaHUiF4775.pngawk–F: ‘{print $1”\t”$3}’ /etc/passwd

以冒号为分隔,打印$1和$3的参数,相当于打印用户名和UID

wKioL1fkimyi2-loAABCTYAZ6rA940.png

tail –3 /etc/fstab |awk ‘{print $2,$4}’

打印/etc/fstab文件倒数三行,每行的第2,4个参数,默认以空格为分隔符

wKioL1fki07SS0V3AAAdDkGFrIQ810.png

awk变量

变量:内置和自定义变量

FS:输入字段分隔符,默认为空白字符

awk -v FS=':' '{print $1,$3,$7}’ /etc/passwd

相当于定义一个变量分隔符为:,输出参数1,3,7

wKioL1fkjaeDXnMDAAA7d_mr758657.png

awk –F: '{print $1,$3,$7}’ /etc/passwd

-v FS=":" 相当于 -F:,所以结果同上图

wKioL1fkjaiBhyYFAAAwvGLmWbE683.png

OFS:输出字段分隔符,默认为空白字符

awk -v FS=‘:’ -v OFS=‘#’ '{print $1,$3,$7}’ /etc/passwd

以冒号为分隔显示第1,3,7的参数,并且替换输出字段的分隔符为#wKioL1fkjzHgqL0hAAA5m2lfLnE859.png

RS:输入记录分隔符,指定输入时的换行符,原换行符仍有效

awk -v RS=“ ” ‘{print }’ /etc/passwd

表示以空格为换行符

wKioL1fkk2ayjtZhAACIHVEHqYs356.png

ORS:输出记录分隔符,输出时用指定符号代替换行符

awk -v RS=“ ” -v ORS='###'‘{print }’ /etc/passwd

相当于把换行符的空格替换成###

wKiom1fkk-HQCWDsAAC9IfDA8-M410.png

NF:字段数量

awk ‘{print NF}’ /etc/fstab , 引用内置变量不用$

打印以空格为分隔符的每行字段数

wKiom1fklz_jkxTpAAAQFp6nNZE001.png

awk -F: '{print $(NF-1)}' /etc/passwd 

NF=7,因为以冒号分隔的参数有7个,$(NF-1)在此处即7-1=6,即显示其家目录的位置

wKiom1fklz6gbnCDAAApnggVUes821.png

NR:行号

awk '{print NR}' /etc/fstab; awk END'{print NR}' /etc/fstab

前半部分相当于打印行号;后半部分相当于打印最后一行的行号,即打印一共有多少行

wKiom1fkmB_wNrfGAAAloB5IY5E396.png

FNR:各文件分别计数,行号

awk '{print FNR}' /etc/fstab /etc/inittab

即先显示完/etc/fstab的各行号,然后从1开始显示第二个文件的各行号

wKiom1fkmWKyJDR3AAAeJ_g2OQY606.png

FILENAME:当前文件名

awk '{print FILENAME}’ /etc/fstab

相当于每一行都替换成文件名

wKioL1fkmhjBcXxJAAA3W9Zxo7k432.png

ARGC:命令行参数的个数

awk '{print ARGC}’ /etc/fstab /etc/inittab ,确认参数是可见ARGV

相当于每一行替换成显示命令行参数

第一个参数是awk

第二个参数是/etc/fstab

第三个参数是/etc/inittab

wKiom1fkmzPxczr_AAASnoS5u_0753.png

awk ‘BEGIN {print ARGC}’ /etc/fstab /etc/inittab

相当于只是在第一行显示参数的个数,

wKioL1fkm5nBfngwAAAR6Rk_tkI208.png

假如少一个/etc/inittab,即参数变成两个

wKiom1fkm93x3nGPAAAL5LiTAUE649.png

ARGV:数组,保存的是命令行所给定的各参数

awk ‘BEGIN {print ARGV[0]}’ /etc/fstab /etc/inittab

awk ‘BEGIN {print ARGV[1]}’ /etc/fstab /etc/inittab

awk ‘BEGIN {print ARGV[2]}’ /etc/fstab /etc/inittab

wKiom1fknQyjxbhZAAAxZFz0fdU597.png

自定义变量

    (1) -v var=value

        变量名区分字符大小写

    (2) 在program中直接定义

示例:

awk -v test='hello gawk' '{print test}' /etc/fstab

相当于每一行都打印变量test,即每一行打印机hello gawk

wKiom1fknemhG0OtAAAyZpFKjIQ371.png

awk -v test='hello gawk' 'BEGIN{print test}'

相当于只在第一行打印变量test

wKioL1fkoBzyEnO0AAARbWjG6i8194.png

awk 'BEGIN{test="hello,gawk";print test

变量直接在中括号里面写,意义同上

wKiom1fkn2fyhpeSAAAPNZqwSpg158.png

printf命令

格式化输出:printf“FORMAT”, item1, item2, …

    (1) 必须指定FORMAT

    (2) 不会自动换行,需要显式给出换行控制符,\n

    (3) FORMAT中需要分别为后面每个item指定格式符

格式符:与item一一对应

    %c: 显示字符的ASCII码

    %d, %i: 显示十进制整数

    %e, %E:显示科学计数法数值

    %f:显示为浮点数

    %g, %G:以科学计数法或浮点形式显示数值

    %s:显示字符串

    %u:无符号整数

    %%: 显示%自身

修饰符:

    #[.#]:第一个数字控制显示的宽度;第二个#表示小数点后精度,%3.1f

    -: 左对齐(默认右对齐)%-15s

    +:显示数值的正负符号%+d

printf示例

awk -F: ‘{printf "%s",$1}’ /etc/passwd

相当于第一个参数是字符窜的时候才显示

wKiom1fkohjhxTRUAAAog_tLeqs265.png

awk -F: ‘{printf "%s\n",$1}’ /etc/passwd

效果如上图,加入“\n”是换行符

wKioL1fkomSTvRbKAAAupkxj9es282.png

awk -F: ‘{printf "Username: %s\n",$1}’ /etc/passwd

显示第一个参数的并且是字符窜的信息,在第一个参数前面加上Username等字符窜信息

wKioL1fku13hytTHAABFagKfisw055.png

awk -F: ‘{printf “Username: %s,UID:%d\n",$1,$3}’/etc/passwd
显示第一个字符串参数和第三个整数参数,并且在其前面加上Username和UID等字符串信息

wKiom1fkvD_TpB8-AAAwR4r6bnw392.png

awk -F: ‘{printf "Username: %15s,UID:%d\n",$1,$3}’/etc/passwd

%15s是多了15个空格的意思,其他同上

wKiom1fkvOWQZmstAAA_0Mc1ywQ522.png

awk -F: ‘{printf "Username: %-15s,UID:%d\n",$1,$3}’/etc/passwd

指与下一个字符隔开15个字符串,参考下图

wKioL1fkvVaQ7vNVAABFtylzOzM035.png

操作符

算术操作符:

    x+y, x-y, x*y, x/y, x^y, x%y

    -x: 转换为负数

    +x: 转换为数值

字符串操作符:没有符号的操作符,字符串连接

赋值操作符:

    =, +=, -=, *=, /=, %=, ^=

    ++, —

比较操作符:

    >, >=, <, <=, !=, ==

模式匹配符:

    ~:左边是否和右边匹配包含

    !~:是否不匹配

模式匹配符示例1: awk –F: '$0 ~ /root/{print $1}‘ /etc/passwd

表示第一个参数包含root等字符的信息

wKioL1fkvr-j8Y-fAAAiHm9wmmk916.png

模式匹配符示例2:awk '$0 !~ /root/‘ /etc/passwd

表示第一个参数不包含root等字符的信息

wKioL1fkvu7CtoHHAAAhZcL5eEg089.png

逻辑操作符:与&&,或||,非!

示例:

awk –F: '$3>=0 && $3<=1000 {print $1}' /etc/passwd

显示UID大于等于0,少于等于1000的的用户名

wKiom1fkv5XwJfXYAAAjG3tsbR0774.png

awk -F: '$3==0 || $3>=1000 {print $1}' /etc/passwd

显示UID等于0或者UID大于等于1000的用户名

wKioL1fkwA7gKQP6AAARgEBpSPA000.png

awk -F: ‘!($3==0){print $1}' /etc/passwd

显示UID不等于0的用户

wKioL1fkwEbRFVj6AAAU8So0Kqk354.png

awk -F: ‘!($3>=500) {print $3}}’ /etc/passwd

显示UID少于500的用户

wKiom1fkwH6gAWUsAAAfCp_SY0g607.png

函数调用:function_name(argu1, argu2, …)

条件表达式(三目表达式):

    selector?if-true-expression:if-false-expression

示例:

awk -F: '{$3>=500?usertype="Common User":usertype="Sysadmin or SysUser";printf"%15s:%-s\n",$1,usertype}' /etc/passwd

判断,如果UID大于等于500,那么usertype=Common User

    否则,那么usertype=Sysadmin or SysUser

wKiom1fk0FjwzmnnAABZDGV47Yc032.png

wKioL1fk0FnB4YZSAABdS-4f3vE573.png

awk PATTERN

PATTERN:根据pattern条件,过滤匹配的行,再做处理,此处支持的是扩展正则表达式

(1)如果未指定:空模式,匹配每一行

(2) /regular expression/:仅处理能够模式匹配到的行,需要用/ /括起来

awk'/^UUID/{print $1}' /etc/fstab

awk'!/^UUID/{print $1}' /etc/fstab

wKiom1fk2C3TPE78AACEDKNMoKM472.png

(3) relational expression: 关系表达式,结果有“真”有“假”,结果为“真”才会被处理

真:结果为非0值,非空字符串

假:结果为空字符串或0值

示例:

awk ‘!0’ /etc/passwd 

非0,即是1,所以打印所有

wKiom1fk1z7hQX3zAABpAnga4bc453.png

awk ‘!1’ /etc/passwd

非1,即0,所以完全不打印

wKioL1fk1z2gC0upAAAJ4RaJJEE504.png

awk –F: '$3>=500{print $1,$3}' /etc/passwd

如果UID>500,就打印用户名和UID

wKioL1fk2QrB4RdmAAAsqYFzcfY654.png

awk -F: '$3<500{print $1,$3}' /etc/passwd

如果UID少于500,就打印用户名和UID

wKioL1fk2xyAwE0_AAAjGdoVgkc988.png

awk -F: '$NF=="/bin/bash"{print $1,$NF}' /etc/passwd

如果最后一个字段等于/bin/bash,就打印用户名和最后一个字段

wKioL1fk24GSFfr-AAA79MjdOmI257.png

awk -F: '$NF==/bash$/{print $1,$NF}' /etc/passwd

(4)line ranges:行范围

startline,endline:/pat1/,/pat2/不支持直接给出数字格式

示例:

awk-F: '/^root\>/,/^nobody\>/{print $1}' /etc/passwd

显示以root开头到以nobody开头的行的第一个参数

wKiom1fk3VWTi4UlAAAvv-AtG48878.png

awk -F: ‘(NR>=10&&NR<=20){print NR,$1}' /etc/passwd

显示大于等于10,少于等于20的行的行号和第一个参数

wKioL1fk3ceyoCSHAAA1B4MVURk524.png

(5) BEGIN/END模式

BEGIN{}: 仅在开始处理文件中的文本之前执行一次

END{}:仅在文本处理完成之后执行一次

awk -F: 'BEGIN{print "line:\tusername"}{(NR>=10&&NR<=20);print NR"\t"$1}' /etc/passwd

wKioL1fk4DTwY0VpAAA7N0_b6T4581.png

awk -F: 'BEGIN{print "line\tusername"}(NR>=10&&NR<=20){print NR"\t"$1}END{print "end file"}' /etc/passwd

wKioL1fk4dLQzhtrAABD_EKDAkw074.png

awk -F: '{print "USER\tUSERID";print $1"\t"$3}END{print "end file"}' /etc/passwd

少了BEGIN以后,每行的开始都打印了USER和USERID

wKioL1fk4mmAEYzIAAA05LqpTLA762.png

awk -F: '{print "USER UID \n—————"}{print $1,$3}' /etc/passwd

wKiom1fk4veBwxIoAAAk1FlDKSk881.png

awk -F: 'BEGIN{print "USER UID \n—————"}{print$1,$3}(NR>=35)

END{print"=============="}' /etc/passwd

wKiom1fk47fhkSKgAAA1QhR_k94822.png

relational expression示例:

真:结果为非0值,非空字符串

假:结果为空字符串或0值

seq 10 |awk ‘i=0’

如果为i=0,那么就不打印

wKiom1fk5cXhBK7PAAAJlzpDinE998.png

seq 10 |awk ‘i=1’

如果i=1,就打印

wKioL1fk5cXwpperAAARonYS8OE445.png

seq 10 | awk 'i=!i‘

如果i不等于i;默认i没有设置任何数值,那么i=0,但是!i,即i不等于0,那么就打印

即第一个数的时候打印,第二个数的时候不打印,

即打印奇数

wKioL1fk5QXwwENUAAAMmgZBQUA321.png

seq 10 | awk '{i=!i;print i}’

从这个例子可以看出上面的例子的打印过程,1 0 1 0…

wKioL1fk5M7jIlRgAAASvpkqWaY669.png

seq 10 | awk ‘!(i=!i)’

打印偶数,参考上面

wKioL1fk5hnghuHXAAAOfuONa0U143.png

seq 10 |awk -v i=1 'i=!i'

给i定义一个初始值1,开始的时候打印条件是i不等于1,即0,所以不打印

然后第二次打印,此时i=0,打印条件是i=!i,即1,所以打印

如此重复,即打印偶数

wKioL1fk5nqCHCSgAAATQgGg4tE333.png

awk action

常用的action分类

    (1) Expressions:算术,比较表达式等

    (2) Control statements:if, while等

    (3) Compound statements:组合语句

    (4) input statements

    (5) output statements:print等

awk控制语句

    { statements;… } 组合语句

    if(condition) {statements;…}

    if(condition) {statements;…} else {statements;…}

    while(conditon) {statments;…}

    do {statements;…} while(condition)

    for(expr1;expr2;expr3) {statements;…}

    break

    continue

    delete array[index]

    delete array

    exit

awk控制语句

if-else

语法:if(condition) statement [else statement]

    if(condition1){statement1}else if(condition2){statement2}else{statement3}

使用场景:对awk取得的整行或某个字段做条件判断

示例:

awk -F: '{if($3>=500)print $1,$3}' /etc/passwd

意义同awk -F: {($3>500)print$1,$3} /etc/passwd

wKioL1fk6LKDDesjAAAxeG-QmPU526.png

awk-F: '{if($NF=="/bin/bash") print $1}' /etc/passwd

显示最后一个参数是/bin/bash的行的第一个参数

wKioL1fk6Z2BgDHQAACEmHZM0sM739.png

awk'{if(NF>5) print $0}' /etc/fstab

显示参数大于5个的所有行

wKiom1fk6iSSbJZDAABx193S6iQ347.png

awk -F: '{if($3>=500) {printf"Common user: %s\n",$1} else {printf"root or Sysuser: %s\n",$1}}' /etc/passwd

awk -F: '{if($3>=500){print "CommonUser:\t"$1}else{print "root or Sysuser:\t"$1}}' /etc/passwd

wKiom1fk7IyxCcTPAABBHkWOBvM526.pngwKioL1fk7HLDxWXgAAAf3BjI6vE051.png

awk -F: '{if($3>=500) printf "Common user: %s\n",$1;else printf "root or Sysuser: %s\n",$1}' /etc/passwd

意义同上,只是格式不同,不同之间可以用(;分号)分隔

wKiom1fk7YfQ0UpjAAArdu4ISdo537.png

df -h | awk -F% '/^\/dev/{print $1}'|awk'$NF>=20{print $1,$5}‘

显示磁盘空间使用率最大的磁盘名称以及其使用的空间

wKioL1fk7nGBR2uKAAARqZ-q3ec990.png

df | awk '/^\/dev\/sd*/{print $5}'| awk -F% '$(NF-1)>=20{print $1}'

显示磁盘空间使用率最大的磁盘使用的空间

wKioL1fk79_C0tJ_AAAQSDdLsqs495.png

awk 'BEGIN{ test=100;if(test>90){print "very good"}else if(test>60){ print "good"}else{print "no pass"}}'

判断test的数值,如果大于90,打印very good,如果大于60,打印good,否则,打印no pass

wKioL1fk8eDRdPB7AABJBjNjNcU734.png

while循环

语法:while (condition) {statement;…}

条件“真”,进入循环;条件“假”,退出循环

使用场景:

    对一行内的多个字段逐一类似处理时使用

    对数组中的各元素逐一处理时使用

示例:

awk '/^[[:space:]]*linux16/{i=1;while(i<=NF){print $i,length($i); i++}}' /etc/grub2.cfg

匹配以空白开头,后面跟着linux16开头的行,默认以空白为分隔符,使i=1,当i小于等于字段数量,就停止,打印每个字段的字段数量。

wKioL1f0-NvT7tr0AABayK1mvro627.png

awk '/^[[:space:]]*linux16/{i=1;while(i<=NF) {if(length($i)>=10) {print $i,length($i)}; i++}}' /etc/grub2.cfg

匹配以空白开头,后面跟着linux16开头的行,默认以空白为分隔符,使i=1,当i小于等于字段数量,就停止,打印大于10个字段数量的字段。

wKiom1f0-NzS9wRHAABNCtiJuLU684.png

do-while循环

语法:do {statement;…}while(condition)

意义:无论真假,至少执行一次循环体

示例:

awk 'BEGIN{ total=0;i=0;do{ total+=i;i++;}while(i<=100);print total}‘

统计1到100的和

wKioL1f0_WSxNxlWAAARaxDD1Bc500.png

思考:下面两语句有何不同?

awk ‘BEGIN{i=0;print ++i,i}’

++i,表示已经+1的i

awk ‘BEGIN{i=0;print i++,i}’

i++,表示0++1,但是状态还是0,下一次的结果才是1

wKiom1f0_e6iANuOAAAUnI9tOmU456.png

for循环

语法:for (expr1;expr2;expr3) {statement;…}

常见用法:

    for(variable assignment;condition;iteration process){for-body}

特殊用法:能够遍历数组中的元素;

    语法:for(varin array) {for-body}

示例:

awk '/^[[:space:]]*linux16/{for(i=1;i<=NF;i++) {print $i,length($i)}}' /etc/grub2.cfg

匹配以空白开头,后面跟着linux16开头的行,默认以空白为分隔符,使i=1,当i小于等于字段数量,就停止,打印每个字段的字段数量。

wKiom1f1AGyh__KZAABckpqk40E287.png

性能比较

执行以下命令对比时间

time (awk 'BEGIN{ total=0;for(i=0;i<=10000;i++){total+=i;};print total;}')

time(total=0;for i in {1..10000};do total=$(($total+i));done;echo $total)

time(for ((i=0;i<=10000;i++));do let total+=i;done;echo $total)

当数值较小的时候,并不会看出明显的差异

wKiom1f1Ar-xmSDCAABQAhX8vj4857.png

当数值变大以后,可以看出,awk计算速度是非常优秀的

wKiom1f1AsWxUG-CAABWG-uZIGU558.png

switch语句

语法:switch(expression) {case VALUE1 or /REGEXP/: statement; case VALUE2 or /REGEXP2/: statement; …; default: statement}

break和continue

break [n] 当匹配对应的n数值时候,终止整个运算

continue [n] 当匹配对应的n数值时候,仅跳过该数值的运算,继续执行下一个数值的运算

awk‘BEGIN{sum=0;for(i=1;i<=100;i++){if(i%2==0){continue}sum+=i}print sum}‘

统计1-100之间的奇数

当i除2余数是0的时候,该数值会被跳过,从而忽略偶数,即只统计奇数

wKiom1f1BZziMNb4AAAoNuzoKEA102.png

awk‘BEGIN{sum=0;for(i=1;i<=100;i++){if(i==66){break}sum+=i}print sum}‘

wKioL1f1Bs7SO0ztAAAmRHeXDcA959.png

next:

提前结束对本行处理而直接进入下一行处理(awk自身循环)

awk -F: '{if($3%2!=0) next; print $1,$3}' /etc/passwd

当遇到符合UID除2余数为非0,就处理下一行,即不处理uid为奇数的行

wKioL1f1ElODBG2fAABUR-bkiv8261.png

以下结果对比以上结果验证想法。

wKioL1f1E9iwQcHLAABRzvSZvrk170.png

awk数组

关联数组:array[index-expression]

index-expression:

(1) 可使用任意字符串;字符串要使用双引号括起来

(2) 如果某数组元素事先不存在,在引用时,awk会自动创建此元素,并将其值初始化为“空串”

若要判断数组中是否存在某元素,要使用“index in array”格式进行遍历

示例:

在awk里面定义一个数组weekdays[“mon”]="Monday“

awk 'BEGIN{weekdays["mon"]="Monday";weekdays["tue"]="Tuesday";print weekdays["mon"]}‘

定义了两个,最后打印数组weekdays的["mon"]项

wKiom1f1FLyDnzf5AAASDmD245s988.png

awk ‘!a[$0]++’ dupfile

若要遍历数组中的每个元素,要使用for循环

for(varin array) {for-body}

注意:var会遍历array的每个索引

示例:

awk 'BEGIN{weekdays["mon"]="Monday";weekdays["tue"]="Tuesday";for(i in weekdays){print weekdays[i]}}'

wKiom1f1MR3A9EmZAAAck2pJmuE552.png

netstat -tan | awk '/^tcp\>/{state[$NF]++}END{for(i in state){print i,state[i]}}'

统计状态出现的次数

wKiom1f1MGrwHi_uAACZl9gA1qQ644.png

awk '{ip[$1]++}END{for(i in ip){print i,ip[i]}}' /var/log/httpd/access_log

统计ip出现的次数

wKioL1f1G3-TJAubAABgbphjssU622.png

awk函数

数值处理:

    rand():返回0和1之间一个随机数

awk 'BEGIN{srand(); for (i=1;i<=10;i++)print int(rand()*100) }'

wKioL1f1JjfSda4AAAAd6XXmRRw099.png

字符串处理:

length([s]):返回指定字符串的长度

sub(r,s,[t]):对t字符串进行搜索r表示的模式匹配的内容,并将第一个匹配的内容替换为s

    echo "2008:08:08 08:08:08" | awk 'sub(/:/,"-",$1)'

wKiom1f1Jtmz0kZ0AAAMwV0L-PA512.png

gsub(r,s,[t]):对t字符串进行搜索r表示的模式匹配的内容,并全部替换为s所表示的内容

    echo "2008:08:08 08:08:08" | awk 'gsub(/:/,"",$1)'

wKiom1f1JzKzec6IAAARzHcstF0658.png

split(s,array,[r]):以r为分隔符,切割字符s,并将切割后的结果保存至array所表示的数组中,第一个索引值为1,第二个索引值为2,…

    netstat -tan | awk '/^tcp\>/{split($5,ip,":");count[ip[1]]++}END{for (i in count) {print i,count[i]}}'

wKiom1f1KC-Q2t_pAACYwqhlgiA077.png


自定义函数

格式:

    function name ( parameter, parameter, … ) {

            statements

            return expression

    }

示例:

    #cat fun.awk

    function max(v1,v2) {

        v1>v2?var=v1:var=v2

        return var

    }

    BEGIN{a=3;b=2;print max(a,b)}

    #awk –f fun.awk

wKioL1f1KW2huMP-AAAjozIx3Ec177.png

awk中调用shell命令

system命令

空格是awk中的字符串连接符,如果system中需要使用awk中的变量可以使用空格分隔,或者说除了awk的变量外其他一律用""引用起来。

    awk BEGIN'{system("hostname")}'

wKioL1f1KwqQUcBXAAAO_PnNF7s914.png

    awk 'BEGIN{score=100; system("echo your score is " score) }'

wKioL1f1KhXjFINKAAASHaNr4EA402.png


awk脚本

将awk程序写成脚本,直接调用或执行

示例:

#cat f1.awk

    {if($3>=1000)print $1,$3}

#awk -F: -f f1.awk /etc/passwd

wKioL1f1LF-DMX90AABLp5y9aOE495.png

#cat f2.awk

    #!/bin/awk -f

    #this is a awk script

    {if($3>=1000)print $1,$3}

#chmod +x f2.awk

#f2.awk –F: /etc/passwd

wKiom1f1LRTxkJ3iAABFIqROeLo666.png

向awk脚本传递参数

格式:

    awk file var=value var2=value2… Inputfile

示例:

    #cat test.awk

        #!/bin/awk –f

        {if($3 >=min && $3<=max)print $1,$3}

    #chmod +x test.awk

    #test.awk -F: min=100 max=200 /etc/passwd

wKioL1f1Ln-BrEe0AAA8Z1AmJ6U436.png

awk综合示例:

1、统计/etc/fstab文件中每个文件系统类型出现的次数

awk '/^UUID/{fs[$3]++}END{for(i in fs){print i,fs[i]}}' /etc/fstab

wKiom1f1LyDSkQEKAAAU6GM1Vh0069.png

2、统计/etc/fstab文件中每个单词出现的次数;

awk '{for(i=1;i<=NF;i++){count[$i]++}}END{for(i in count){print i,count[i]}}' /etc/fstab

wKioL1f1L7nQs571AABZ4JL1JCE957.png

3、分别统计1和2班的平均成绩

   score.txt

name    score   class

zhu    81     1

wang   90     2

zh     88     1

w      99     2

wKiom1f1riTig_x8AAAy2kPShWI989.png

原创文章,作者:~微风~,如若转载,请注明出处:http://www.178linux.com/48689

(1)
上一篇 2016-10-05 22:42
下一篇 2016-10-06 13:15

相关推荐

  • 以后的运维是linux系统的天下

       搞IT行业差不多10年了,也接触过网络设备的配置,比如防火墙和交换机的管理,也管理过虚拟化平台,比如vmware的虚拟化,还懂一些简单的oracle数据库的安装,存储、光钎交换机的配置等,因为曾经在传统企业做过8年多的网管,当时的企业规模也是比较大,也接触了很多设备,但就是linux接触的不是很多,后来去一家公司做系统集成,也是什么…

    Linux干货 2016-10-19
  • MySQL备份与恢复

    备份与恢复 数据备份就是将数据以某种方式加以保留,以便在系统需要时重新恢复和利用。其作用主要体现在如下两个二方面: 在数据遭到意外事件破坏时,通过数据恢复还原数据 数据备份是历史数据保存归档的最佳方式 数据恢复就是把遭到破坏、删除和修改的数据还原为可使用的数据的过程 为什么要备份数据? 在生产环境中我们数据库可能会遭遇各种各样的不测从而导致数据丢失, 大概分…

    Linux干货 2016-11-25
  • 虚拟化网络之OpenvSwitch(三)

    上一篇介绍了openvswitch利用GRE协议,搭建多台宿主机的虚拟网络,接下来在利用vxlan通道搭建一个跨多宿主机的虚拟化网络,深入了解openvswitch的功能。 一、实验拓扑 ip地址分配:  A1:192.168.10.1/24  A2:192.168.10.10/24   B1:192.168.10.2…

    系统运维 2016-03-27
  • 软件包管理

    软件包管理 在linux中有人多的软件使用,比如我们使用的ifconfig,fdisk,btrfs文件系统等。可能这些软件在你原有的linux系统中并不存在,那么我们如何去使用这些工具呢。我们知道windows在需要软件的时候需要安装,当然对于linux系统一样需要安装。在linux下常用的安装工具我们使用的是rpm程序包(当然我们这边是针对CentOS而言…

    Linux干货 2016-08-19
  • linux 启动文件 /etc/fstab 文件删除 无法开机 修复的方法

    /etc/fstab文件删除系统自检无法挂载 无法启动 按照提示进入命令行  手动写/etc/fstab,如无内核文件  无grub  就缺啥补啥 fdisk    -l 查看当前分区  blkid    查看分区文件系统的卷标 手动挂载 …

    Linux干货 2016-09-13