awk腳本編程實例講解(判斷,循環,數組)

1.條件判斷

if語句格式:{if(表達式) {語句;語句;...}}

統計系統用戶數

0-10001系統用戶,大於1000普通用戶

#awk -F: '{if($3>0 && $3<1000){count++;}} END{print count}' /etc/passwd

記住:awk是逐行處理。所以對每一行進行判斷處理後進行加1操作。

打印普通用戶

[root@bigdata3 dan_test]# awk -F : '{if($3 >1000){print $1}}' /etc/passwd

統計普通用戶數

[root@bigdata3 dan_test]# awk -F : '{if($3 >1000){ i++}} END{print i}' /etc/passwd

if ...else 語句格式:{if(表達式) {語句;語句;...} else{語句;語句;...}}

  • #awk -F:'{if ($3 == 0){print $1} else {print $7}}' /etc/passwd
  • #awk -F:'{if ($3 == 0){count++} else {i++}}' /etc/passwd
  • #awk -F:'{if ($3 == 0){count++} else {i++}} END{print "管理員個數:"count;print"系統用戶數:"i }' /etc/passwd

統計普通用戶個數,系統用戶個數,管理員的個數

[root@bigdata3 dan_test]# awk -F : '{if($3==0){i++} else if($3>0 && $3<100){j++} else if($3>1000){k++}} END{print "管理員用戶數:"i;print"系統用戶數:"j;print"普通用戶數"k}' /etc/passwd

2.循環

2.1 for循環C風格for

[root@bigdata3 dan_test]# awk 'BEGIN{for(i=1;i<5;i++){print i}}'

[root@bigdata3 dan_test]# awk -F: '{for(i=1;i<=5;i++){print $0}}' awk.txt

cat awk.txt

dandan:dandan1

[root@bigdata3 dan_test]# awk -F: '{for(i=1;i<=NF;i++){print $i}}' awk.txt

NF:表示列的個數

2.2 whiel循環

cat password

[root@bigdata3 dan_test]# awk -F : '/^root/{i=1;while(i<8){print $i;i++}}' password

 

[root@bigdata3 dan_test]# awk -F : '{i=1;while(i<NF){print $i;i++}}' password

解釋:是針對每一行,循環該行按照:切割後的次數,並將該字段打印出來。

主要理解點:針對每一行進行循環,理解的時候注意對awk的工作原理的理解

[root@bigdata3 dan_test]# awk -F : '{i=1;while(i<10){print $0;i++}}' password

表示將每行打印10次

cat b.txt

[root@bigdata3 dan_test]# awk '{i=1;while(i<=NF){print $i;i++}}' b.txt

 

實際上是對每行的內容進行遍歷

awk默認分割是對空行的分割

  • (1)對每一行進行循環,第一行循環2次,分別打印每個字段
  • (2)第二行循環3次,分別打印每個字段
  • (3)第三行循環4次,分別打印每個字段

因此實際上是對每行內容的遍歷

3. awk數組

數組:存儲一系列相同類型的元素,健/值方式存儲,通過下標(健)來訪問值。awk中數組稱爲關聯數組,不僅可以使用數字作爲下標,還可以使用字符串作爲下標。數組元素的鍵和值存儲在 awk 程序內部的一個表中,該表採用散列算法,因此數組元素是隨機排

序。

數組格式:array[index]=value

數據準備:

[root@bigdata3 dan_test]# tail -n5 /etc/passwd

[root@bigdata3 dan_test]# tail -n5 /etc/passwd | awk -F: '{username[++i]=$1} END{print username[i]}'

分析:該數組定義是將每行的第一個字段賦值給數組username,此時i沒操作一次進行加一,到最後打印的是行處理結束時所賦值的字段,由於只有5行,所以打印username[5] flume。END是行處理完了才執行。

注意:++i數組的下標是先從1開始,i++數組的下標是先從0開始。

那麼如果想打印數組中每個元素呢?此時需要for循環在END語句中進行遍歷

[root@bigdata3 dan_test]# tail -n5 /etc/passwd | awk -F: '{username[++i]=$1} END{for (i in username){print username[i],i}}'

下標從i++開始注意變化:

[root@bigdata3 dan_test]# tail -n5 /etc/passwd | awk -F: '{username[i++]=$1} END{for (i in username){print username[i],i}}'

統計/etc/passwd下各類型shell的數量

cat /etc/shell

[root@bigdata3 dan_test]# awk -F: '{shells[$NF]++} END{for(i in shells){print i,shells[i]}}' /etc/passwd

知識點準備:

  • (1)數組默認初始值是0
  • (2)a[i]++表示對數組元素值自加
  • (3)shells[$NF]:表示把每行的最後一個字段作爲索引,同時其值自加

推理過程:

  • shell[/bin/bash]:當讀取第一行數據時候遇到/bin/bash此時其數組對應的值加一,數組值爲1
  • 當讀取第二行數據遇到/sbin/nologin,此時shell[/sbin/nologin]其對應的數組值加一,數組值爲1
  • 當讀取第三行數據時,遇到/bin/bash,數組值shell[/bin/bash]加一,此時對應值爲2
  • 當讀取第四行數據時,遇到/sbin/nologin,此時shell[/sbin/nologin]其對應的數組值加一,數組值爲2
  • ......
  • 依次類推
  • 最終直到所有的行被讀完,數組中保存的累計值就爲該字符串索引出現的次數
  • 推理時候一定要聯想讀取數據是一行一行處理和一般數組區別開來。
  • 遍歷最終結果的時候,類似於數學中的反函數,可以把字符串作爲索引的這類數組問題理解爲反函數。這樣給字符串統計帶來了方便,即awk中shell[字符串]++這樣的數組用來統計某個文件中字符串出現的次數。也就是你想統計誰,讓誰作爲索引然後循環遍歷就可以了。

統計 TCP 連接狀態

[root@bigdata3 dan_test]# netstat -antp |awk '/^tcp/{a[$6]++}END{for(v in a)print a[v],v}'

打印/etc/services出現次數大於等於 2 的

[root@bigdata3 dan_test]# tail /etc/services |awk '{a[$1]++}END{for(v in a) if(a[v]>=2){print a[v],v}}'

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章