11. [轉載]linux中awk命令使用
awk是一個非常棒的數字處理工具。相比於sed常常作用於一整行的處理,awk則比較傾向於將一行分為數個“欄位”來處理。執行效率高,而且程式碼簡單,對格式化的文字處理能力超強。先來一個例子:
檔案a,統計檔案a的第一列中是浮點數的行的浮點數的平均值。用awk來實現只需要一句話就可以搞定
$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
(分析:$1~/^[0-9]+\.[0-9]*/表示$1與“/ /”裡面的正規表示式進行匹配,若匹配,則total加上$1,且len自增,即數目加1. “^[0-9]+\.[0-9]*”是個正規表示式,“^[0-9]”表示以數字開頭,“\.”是轉義的意思,表示“.”為小數點的意思。“[0-9]*”表示0個或多個數字)
awk的一般語法格式為:
awk [-引數 變數] 'BEGIN{初始化}條件型別1{動作1}條件型別2{動作2}。。。。END{後處理}'
其中:BEGIN和END中的語句分別在開始讀取檔案(in_file)之前和讀取完檔案之後發揮作用,可以理解為初始化和掃尾。
(1)引數說明:
-F re:允許awk更改其欄位分隔符
-v var=$v 把v值賦值給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語法的程式檔案。
(2)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
(3)awk讀取shell中的變數
可以使用-v選項實現功能
$b=1
$cat f
apple
$awk -v var=$b '{print var, $var}' f
1 apple
至於有沒有辦法把awk中的變數傳給shell呢,這個問題我是這樣理解的。shell呼叫awk實際上是fork一個子程式出來,而子程式是無法向父程式傳遞變數的,除非用重定向(包括管道)
a=$(awk '{print $b, '$b'}' f)
$echo $a
apple 1
(4)輸出重定向
awk的輸出重定向類似於shell的重定向。重定向的目標檔名必須用雙引號引用起來。
$awk '$4 >=70 {print $1,$2 > "destfile" }' filename
$awk '$4 >=70 {print $1,$2 >> "destfile" }' filename
(5)awk中呼叫shell命令:
1)使用管道
awk中的管道概念和shell的管道類似,都是使用"|"符號。如果在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 (解釋:/west/{count++}表示與“wes”t進行匹配,若匹配,則count自增)
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
(5)awk結合match使用,提取匹配字串:
檔案內容:
hostname
0.0.0.0
要求提取0.0.0.0
命令:
cat filename | grep -A 1 "hostname" |awk 'match($0, "(.*)", a) {print a[1]}'
其中 grep -A n,表示多往下輸出n行
相關文章
- Linux中awk命令詳解Linux
- 【案例】Linux三劍客中awk命令如何使用?Linux
- linux系統awk命令使用詳解Linux
- linux的awk命令Linux
- Linux命令之grep/sed/awk等行轉列Linux
- Linux:“awk”命令的妙用Linux
- Linux awk 命令詳解Linux
- Linux awk命令中如何刪除陣列Linux陣列
- Linux系統中的管道命令、grep命令、sed命令和awk命令Linux
- 技能篇:awk教程-linux命令Linux
- awk命令使用例項
- 使用 awk 命令統計文字
- 【Linux篇】--awk的使用Linux
- linux awk使用方法Linux
- Linux awk使用案例教程Linux
- 007 Linux 命令三劍客之-awkLinux
- awk命令
- Linux (三劍客之三) awk命令詳解Linux
- 程式猿必知必會Linux命令之awkLinux
- 5分鐘帶你瞭解Linux awk命令!Linux
- Linux 三劍客 Awk、Sed、Grep 命令詳解Linux
- Linux三劍客Awk、Sed、Grep 命令詳解Linux
- Linux中 awk命令根據列的索引批次提取列的資料Linux索引
- linux 中awk命令實現按照 指定的字元對文字進行排序Linux字元排序
- Linux 中 awk命令如何擷取指定欄位的前幾個字元Linux字元
- Linux awk 中 xxxx的應用Linux
- 寶付揭秘Linux支付命令操作之grep、sed、awkLinux
- Linux中Type命令如何使用Linux
- Linux 中如何使用 id 命令Linux
- Linux中如何使用alias命令Linux
- Linux中awk命令實現指定欄位的第一個字母大寫Linux
- 在Linux中,如何使用awk和sed進行文字處理?Linux
- linux之 awkLinux
- Linux-(awk)Linux
- linux之awkLinux
- 在 Linux 命令列中轉換大小寫Linux命令列
- Linux轉包命令Linux
- Linux中printf命令使用例項Linux
- linux中xargs命令的使用方式Linux