AWK详细用法

awk非常的优秀,运行效率高,而且代码简单,对格式化的文本处理能力超强。基本上grep和sed能干的活awk全部都能干,而且干得更好。

先来一个很爽的例子:
文件a,统计文件a的第一列中是浮点数的行的浮点数的平均值。用awk来实现只需要一句话就可以搞定(当然,这个东东用python也可以很轻松的实现,只是无论如何都得新建一个文件;别妄想用bash
shell来做,那可是浮点数!!!)
$cat a
1.021 33
1#.ll   44
2.53
6
ss    7

awk ‘BEGIN{total = 0;len = 0}
{if($1~/^[0-9]+\.[0-9]*/){total += $1; len++}} END{print total/len}‘
a

niubility!

awk的语法:
awk [-F re]
[parameter...] [‘program‘] [-f ‘programfile‘] [in_file_list]

awk里面的BEGIN,END结构:
BEGIN和END中的语句分别在开始读取文件(in_file)之前和读取完文件之后发挥作用,可以理解为初始化和扫尾。

awk里面的if..else;   while ; do..while; for; break;
continue; printf 语法都和C语言的语法一致;而且awk支持使用if (key in
array)这样的判断语句(其中,array是数组,这一点和python的语法非常相像。);awk支持使用for (key in
array)这样的语法来遍历数组(也是和python的语法很相像。)

我会用的参数的说明:
-F
re:允许awk更改其字段分隔符
-v var=val
把val值赋值给var(变量通信的好方法啊~~今天才知道这个选项,想想之前写的代码,抓狂啊~~)如果有多个变量要赋值,那么就写多个-v,每个变量赋值对应一个-v
e.g.
要打印文件a的第num行到num+num1行之间的行, awk -v num=$num -v num1=$num1
‘NR==num,NR==num+num1{print}‘ a
-f
progfile:允许awk调用并执行progfile程序文件,当然progfile必须是一个符合awk语法的程序文件

我会用的awk内置变量:
ARGC    命令行参数的个数
ARGV:命令行参数数组
ARGIND 当前被处理文件的ARGV标志符
e.g
有两个文件a 和b
awk ‘{if(ARGIND==1){print "处理a文件"} if(ARGIND==2){print "处理b文件"}}‘ a
b
文件处理的顺序是先扫描完a文件,再扫描b文件

NR 已经读出的记录数
FNR   当前文件的记录数
上面的例子也可以写成这样:
awk
‘NR==FNR{print "处理文件a"} NR > FNR{print "处理文件b"}‘ a
b
输入文件a和b,由于先扫描a,所以扫描a的时候必然有NR==FNR,然后扫描b的时候,FNR从1开始计数,而NR则接着a的行数继续计数,所以NR
> FNR

e.g 要显示文件的第10行至第15行
awk ‘NR==10,NR==15{print}‘ a

FS 输入字段分隔符(缺省为:space:),相当于-F选项
awk -F
‘:‘ ‘{print}‘ a    和   awk
‘BEGIN{FS=":"}{print}‘ a 是一样的

OFS输出字段分隔符(缺省为:space:)
awk -F ‘:‘
‘BEGIN{OFS=";"}{print $1,$2,$3}‘ b
如果cat
b为
1:2:3
4:5:6
那么把OFS设置成";"后就会输出
1;2;3
4;5;6
(小注释:awk把分割后的第1、2、3个字段用$1,$2,$3...表示,$0表示整个记录(一般就是一整行))

NF:当前记录中的字段个数
awk -F ‘:‘ ‘{print NF}‘
b的输出为
3
3
表明b的每一行用分隔符":"分割后都3个字段
可以用NF来控制输出符合要求的字段数的行,这样可以处理掉一些异常的行
awk
-F ‘:‘ ‘{if (NF == 3)print}‘ b

RS:输入记录分隔符,缺省为"\n"
缺省情况下,awk把一行看作一个记录;如果设置了RS,那么awk按照RS来分割记录
例如,如果文件c,cat
c为
hello world; I want to go swimming tomorrow;hiahia
运行 awk ‘BEGIN{ RS =
";" } {print}‘ c 的结果为
hello world
I want to go swimming
tomorrow
hiahia

合理的使用RS和FS可以使得awk处理更多模式的文档,例如可以一次处理多行,例如文档d cat
d的输出为
1 2
3 4 5

6 7
8 9 10
11 12

hello
每个记录使用空行分割,每个字段使用换行符分割,这样的awk也很好写
awk ‘BEGIN{ FS =
"\n"; RS = ""} {print NF}‘ d 输出
2
3
1

ORS:输出记录分隔符,缺省为换行符,控制每个print语句后的输出符号
awk
‘BEGIN{ FS = "\n"; RS = ""; ORS = ";"} {print NF}‘ d
输出
2;3;1

awk的数组:

awk的数组是一个很值得一说的东东。awk的数组从行为上看的话更像关联数组,或者说map、字典,或者说散列。awk的数组接受字符串下标,并接受速度很快的in查询
文件e是由小写的字母组成,cat
e 输出为
a
b
z
...
如果要统计不同的字母出现的个数,那么可以使用数组来实现
awk ‘{arr[$0]++}
END{ for (key in arr) print key, "-->",arr[key] }‘ e
使用for( key in
arr)来遍历数组的时候,输出的次序是不可预测的,这一点跟python的字典遍历是一致的。
在gawk中,可以使用asort内置函数实现数组的排序,其他的awk版本中还没有发现有类似的排序函数。一个折中的办法是先awk完再用管道传给sort来排序。sort使用-k选项可以控制使用指定列排序。

awk的多维数组:

awk的多维数组在本质上是一维数组,更确切一点,awk在存储上并不支持多维数组。awk提供了逻辑上模拟二维数组的访问方式。例如,array[2,4]
= 1这样的访问是允许的。awk使用一个特殊的字符串SUBSEP
(\034)作为分割字段,在上面的例子中,关联数组array存储的键值实际上是2\0344。

类似一维数组的成员测试,多维数组可以使用 if ( (i,j) in
array)这样的语法,但是下标必须放置在圆括号中。
类似一维数组的循环访问,多维数组使用 for ( item in array
)这样的语法遍历数组。与一维数组不同的是,多维数组必须使用split()函数来访问单独的下标分量。split ( item, subscr,
SUBSEP)

awk读取shell中的变量
可以使用-v选项实现功能
     $b=1
     $cat
f
     apple

$awk -v var=$b ‘{print var,
$var}‘ f
1
apple

除了使用-v选项外,还可以使用"‘$variable‘"的方式从shell往awk传递变量(注意:这里是单引号)
$awk
‘{print $b, ‘$b‘}‘ f
apple 1

至于有没有办法把awk中的变量传给shell呢,这个问题我是这样理解的。shell调用awk实际上是fork一个子进程出来,而子进程是无法向父进程传递变量的,除非用重定向(包括管道)
$a=$(awk
‘{print $b, ‘$b‘}‘ f)
$echo $a
apple 1

getline

getline为awk所提供的输入指令.

其语法如下 :














语法

由何处读取数据

数据读入后置于

getline var < file

所指定的 file

变量 var(var省略时,表示置于$0)

getline var

pipe 变量

变量 var(var省略时,表示置于$0)

$awk ‘BEGIN{ "date" | getline d; close("date");print d}‘
f
Sun Nov 9 20:55:12 CST 2008
$awk ‘BEGIN{getline name < "/dev/tty"}

$awk ‘BEGIN{while(getline < "/etc/passwd" > 0) { lc++ }; print lc }‘
f
只要getline的返回值大于0,即读入一行,循环就会继续。
getline如果如成功读取,返回1,否则返回-1,如果遇到EOF,则返回0。getline在读取的同时会设置NF,NR,FNR等内置变量
如果getline后没有变量,则默认置于$0
$awk
‘BEGIN{ while(("ls" | getline) > 0) print}‘ f
(以上3个例子来自UNIX Shells
By Example
 Fourth Edition, Section 6.26.4)

输出重定向

awk的输出重定向类似于shell的重定向。重定向的目标文件名必须用双引号引用起来。
$awk ‘$4 >=70
{print $1,$2 > "destfile" }‘ filename
$awk ‘$4 >=70 {print $1,$2
>> "destfile" }‘ filename

awk中调用shell命令:

1)使用管道
awk中的管道概念和shell的管道类似,都是使用"|"符号,在上面getline中{"date"
| getline
d;}就是使用了管道。如果在awk程序中打开了管道,必须先关闭该管道才能打开另一个管道。也就是说一次只能打开一个管道。shell命令必须被双引号引用起来。“如果打算再次在awk程序中使用某个文件或管道进行读写,则可能要先关闭程序,因为其中的管道会保持打开状态直至脚本运行结束。注意,管道一旦被打开,就会保持打开状态直至awk退出。因此END块中的语句也会收到管道的影响。(可以在END的第一行关闭管道)”
awk中使用管道有两种语法,分别是:
awk
output | shell input
shell output | awk input

对于awk output | shell
input来说,shell接收awk的输出,并进行处理。需要注意的是,awk的output是先缓存在pipe中,等输出完毕后再调用shell命令处理,shell命令只处理一次,而且处理的时机是“awk程序结束时,或者管道关闭时(需要显式的关闭管道)”
$awk
‘/west/{count++} {printf "%s %s\t\t%-15s\n", $3,$4,$1 | "sort +1"} END{close
"sort +1"; printf "The number of sales pers in the western"; printf "region is "
count "." }‘
datafile
printf函数用于将输出格式化并发送给管道。所有输出集齐后,被一同发送给sort命令。必须用与打开时完全相同的命令来关闭管道(sort
+1),否则END块中的语句将与前面的输出一起被排序。此处的sort命令只执行一次。

在shell output | awk
input中awk的input只能是getline函数。shell执行的结果缓存于pipe中,再传送给awk处理,如果有多行数据,awk的getline命令可能调用多次。
$awk
‘BEGIN{ while(("ls" | getline d) > 0) print d}‘ f

2)使用system命令

$awk ‘BEGIN{system("echo
abc")}‘
需要注意的是system中应该使用shell命令的对应字符串。awk直接把system中的内容传递给shell,作为shell的命令行。

3)system命令中使用awk的变量
空格是awk中的字符串连接符,如果system中需要使用awk中的变量可以使用空格分隔,或者说除了awk的变量外其他一律用""引用起来。
$awk
‘BEGIN{a = 12; system("echo " a) }‘

还有好多呀,以后再补充
awk的运算符
next等函数
更多的输入输出(输出到多个文件,关闭文件,输出到命令)
awk的内置函数:

gsub, index, length, match, printf, split, sprintf, substr,
tolower, toupper, atan, cos, exp, int, log, rand, sin, sqrt, srand,
system

AWK详细用法,码迷,mamicode.com

时间: 2024-10-16 23:46:13

AWK详细用法的相关文章

Linux文本处理三剑客——awk详细用法

awk是行处理器: 相比较屏幕处理的优点,在处理庞大文件时不会出现内存溢出或是处理缓慢的问题,通常用来格式化文本信息.awk处理过程: 依次对每一行进行处理,然后输出 . awk命令形式:awk [-F|-f|-v] 'BEGIN{} //{command1; command2} END{}' file [-F|-f|-v] 大参数,-F指定分隔符,-f调用脚本,-v定义变量 var=value BEGIN 初始化代码块,在对每一行进行处理之前,初始化代码,主要是引用全局变量,设置FS分隔符 E

Display:Block 详细用法

根据CSS规范的规定,每一个网页元素都有一个display属性,用于确定该元素的类型,每一个元素都有默认的display属性值,比如div元素,它的默认display属性值为"block",成为"块级"元素(block-level):而span元素的默认display属性值为"inline",称为"行内"元素. 块级元素: 动占据一定矩形空间,可以通过设置高度.宽度.内外边距等属性,来调整的这个矩形的样子: 行内元素: 自己的

DOM Style样式对象的详细用法

DOM Style样式对象的详细用法 HTML Style样式比较复杂,相应访问.修改方法也有所差异.参考相关资料,整理如下. 典型Html文件如下,有三种定义方式. <head>     <style type="text/css">                /* 内部样式 */       h3 {color:green;}     </style>             <!-- 外部样式 style.css -->    

文件/目录权限设置命令chmod的详细用法

chmod是文件/目录权限设置的命令,在Linux中经常遇到,本博文以下总结chmod的详细用法. Linux/Unix的档案调用权限分为三级,即档案拥有者user.群组group.其他other.u表示该档案的拥有者,g表示与该档案的拥有者属于同一个群体(group)者,o表示其他以外的人,a表示这三者皆是. + 表示增加权限.- 表示取消权限.= 表示唯一设定权限. r表示可读取,w表示可写入,x表示可执行. 举例说明: (1).将档案file1.txt 设为所有人皆可读取: chmod u

mysql中游标在存储过程中的详细用法

昨天写的一个东东,分享下给大家. drop PROCEDURE  if exists sp_cleanUserData; CREATE  PROCEDURE `sp_cleanUserData`() BEGIN /*定义游标*/ declare v_dt bigint(20) default 0 ; declare v_num INT DEFAULT 0; /*游标循环到末尾时给定义的常量赋值*/ declare cur_userId   CURSOR FOR select  userId fr

awk基本用法

RS,ORS,FS,OFS,NR,NF,$0,$n RS 输入的行分隔号 ORS 输出的行分隔号 FS 输入的列分隔号 OFS 输出的列分隔号 NR 行号 NF 尾列号 $0 表示所有列(整行) $n 表示第n列 例如: shell> cat aa 1 2 3 4 5 6 shell> awk 'BEGIN{ORS=",\n"}{print}' aa 1 2, 3 4, 5 6, RS反之 shell> awk 'BEGIN{ORS=",\n"}{

Linux sed 和 awk的用法

sed用法: 原文链接:http://www.cnblogs.com/dong008259/archive/2011/12/07/2279897.html sed是一个很好的文件处理工具,本身是一个管道命令,主要是以行为单位进行处理,可以将数据行进行替换.删除.新增.选取等特定工作,下面先了解一下sed的用法sed命令行格式为:         sed [-nefri] 'command' 输入文本 常用选项:        -n∶使用安静(silent)模式.在一般 sed 的用法中,所有来自

Linux中find、grep命令详细用法

在linux下面工作,有些命令能够大大提高效率.本文就向大家介绍find.grep命令,他哥俩可以算是必会的linux命令,我几乎每天都要用到他们.本文结构如下: find命令 find命令的一般形式 find命令的常用选项及实例 find与xargs grep命令 grep命令的一般形式 grep正则表达式元字符集(基本集) grep命令的常用选项及实例 1.find命令 find命令是一个无处不在命令,是linux中最有用的命令之一.find命令用于:在一个目录(及子目录)中搜索文件,你可以

SplitContainer 控件详细用法(转)

1.可以将 Windows 窗体 SplitContainer 控件看作是一个复合体,它是由一个可移动的拆分条分隔的两个面板.当鼠标指针悬停在该拆分条上时,指针将相应地改变形状以显示该拆分条是可移动的.使用 SplitContainer 控件,可以创建复合的用户界面(通常,在一个面板中的选择决定了在另一个面板中显示哪些对象).这种排列对于显示和浏览信息非常有用.拥有两个面板使您可以聚合不同区域中的信息,并且用户可以轻松地使用拆分条(也称为"拆分器")调整面板的大小.另外,还可以嵌套多个