shell脚本四剑客之awk详解

news2025/1/10 14:09:17

文章目录

    • awk的介绍
    • awk能够干什么
    • awk的格式
    • 工作原理:
    • 记录和域
      • 内建变量的用法
        • 1. FS
        • 2. OFS
        • 3.RS
        • 4. ORS
        • 5. NF
        • 6. NR
    • BEGIN 和END语句块
    • 常见案例
      • 1. 使用NR行号提取ip
      • 2. 打印UID小于10的账号名称和UID信息
      • 3. 数学运算
      • 4. AWK打印硬盘设备名称,默认以空格为分割:
      • 5. AWK以空格、冒号、\t(Tab缩进) 、分号为分割显示第4列内容:
      • 6. AWK以冒号分割,打印第一列,同时将内容追加到/tmp/awk.log下:
      • 7. 打印test.txt文件中的第3行至第5行(2种写法):NR表示打印行,$0表示文本所有域:
      • 8. 打印test.txt文件中的第3行至第5行的第一列与最后一列:
      • 9. 打印test.txt文件中,长度大于80的行号:
      • 10. AWK引用Shell变量,使用-v或者双引号+单引号即可:
    • 总结

awk的介绍

shell脚本中最难的不是shell的语法难,而是在shell编程这门语言中,又突然出来一门编程语言,这个编程语言叫: awk

awk以用来处理数据和生成报告(excel);
处理的数据可以是一个或多个文件;可以是直接来自标准输入,也可以通过管道获取标准输入;

awk可以在命令行上直接编辑命令进行操作,也可以编写成awk程序来进行更为复杂的运用。

awk能够干什么

学习具体使用前,先来看下 awk 能干些什么事情:

  1. 能够将给定的文本内容,按照我们期望的格式输出显示,打印成报表。
  2. 分析处理系统日志,快速地分析挖掘我们关心的数据,并生成统计信息;
  3. 方便地用来统计数据,比如网站的访问量,访问的 IP 量等;
  4. 通过各种工具的组合,快速地汇总分析系统的运行信息,让你对系统的运行了如指掌;
  5. 强大的脚本语言表达能力,支持循环、条件、数组等语法,助你分析更加复杂的数据;
    6
    awk 不是万能的,它比较擅长处理格式化的文本,比如 日志、csv 格式数据等;

awk的格式

简单结构:
awk [选项] ‘模式 {动作}’ 输入文件

awk [options] ‘pattern{attion}’ file

  • 模式即pattern,可以类似理解成sed的模式匹配,可以由表达式组成,也可以是两个正斜杠之间的正则表达式。比如NR==1,这就是模式,可以把它理解为一个条件。

  • 动作即action,是由在大括号里面的一条或多条语句组成,语句之间使用分号隔开。

详细结构

awk ‘BEGIN{ print “start” } pattern{ commands } END{ print “end” }’ filename

一个 awk 脚本通常由 BEGIN 语句 + 模式匹配 + END 语句三部分组成,这三部分都是可选项

工作原理:

简单版:
第一步执行 BEGIN 语句
第二步从文件或标准输入读取一行,然后再执行 pattern 语句,逐行扫描文件到文件全部被读取
第三步执行 END 语句

在这里插入图片描述

详细执行流程:

  • 首先执行关键字 BEGIN 标识的 {} 中的命令;
  • 完成 BEGIN 大括号中命令的后,开始执行 body 命令;
  • 逐行读取数据,默认读到 \n 分割的内容为一条 记录,其实就是行的概念;
  • 将记录按照指定的分隔符划分为 字段,其实就是列的概念;
  • 循环执行 body 块中的命令,每读取一行,执行一次 body,最终完成 body 执行;
  • 最后,执行 END 命令,通常会在 END 中输出最后的结果;

awk 是输入驱动的,有多少输入行,就会执行多少次 body 命令

我们在学习过程中,要时刻记着:

记录 (Record) 就是行
字段 (Field) 就是列
BEGIN 是预处理阶段
body 是 awk 真正工作的阶段
END 是最后处理阶段。

案例: 输出test.txt中的每一行

[root@itlaoxin41 ~]# awk '{print $0}' test.txt
cccc
old new old now 

在这个案例中: awk每次读取一行
具体的执行过程:

  • 读取文件第一行 (awk 默认按行读取文件)
  • 将所读取的行赋值给 awk 的变量 $0,于是 $0 中保存的就是本次所读取的行数据
  • 进入代码块 {print $0} 并执行其中代码 print $0,即输出 $0,也即输出当前所读取的行
  • 执行完本次代码之后,进入下一轮 awk 循环:继续读取下一行 (第二行)


全部读取完成后,退出awk

记录和域

域标记: awk执行时,其浏览标记为$1, 2... 2... 2...n,这种方法称为域标记

$1 表示参照第一域, 多个域用,分割,比如$1,$3
$0 表示所有域

内置变量:

···

NR:表示当前的行数;
NF:表示当前的列数;
RS:行分隔符,默认是换行;
FS:列分隔符,默认是空格和制表符;
OFS:输出列分隔符,用于打印时分割字段,默认为空格
ORS:输出行分隔符,用于打印时分割记录,默认为换行符

案例:

以冒号为分隔符,打印第三个域

[root@itlaoxin41 ~]# awk -F ":" '{print $3}' /etc/passwd

假设我们想截取多列,比如:

截取用户名,用户ID,家目录和bash

[root@itlaoxin41 ~]# awk -F ":" '{print $1,$3,$6,$7}' /etc/passwd
root 0 /root /bin/bash

这里需要知道,awk默认以空格为分隔符

内建变量的用法

1. FS

指定列分隔符
···

[root@itlaoxin41 ~]# echo "111|222|333" |awk '{print $1}'
111|222|333
[root@itlaoxin41 ~]# echo "111|222|333" |awk 'BEGIN{FS="|"}{print $1}'
111
[root@itlaoxin41 ~]# 

2. OFS

输出列分隔符,用于打印时分割字段,默认为空格

[root@itlaoxin41 ~]# echo "111 222 333" |awk 'BEGIN{OFS="/"}{print $1,$2}'
111/222
[root@itlaoxin41 ~]# echo "111 222 333" |awk 'BEGIN{OFS="/"}{print $1,$2,$3}'
111/222/333
[root@itlaoxin41 ~]# 

3.RS

行分隔符,默认是换行符;

[root@itlaoxin41 ~]# echo "111 |222|333|444" |awk 'BEGIN{RS="|"}{print $0}'
111 
222
333
444

4. ORS

ORS指定输出行分隔符

[root@itlaoxin41 ~]# echo "111 |222|333|444" |awk 'BEGIN{RS="|"}{print $0}'>test.sh

[root@itlaoxin41 ~]# cat test.sh
111 
222
333
444

[root@itlaoxin41 ~]# awk 'BEGIN{ORS="|";}{print $0}' test.sh
111 |222|333|444||

5. NF

当前的列数

[root@itlaoxin41 ~]# cat test.sh 
111 
222
333
444

[root@itlaoxin41 ~]# awk '{print NF}' test.sh
1
1
1
1
0

6. NR

当前行的行数

[root@itlaoxin41 ~]# cat test.sh 
111 
222
333
444

[root@itlaoxin41 ~]# awk '{print NR}' test.sh
1
2
3
4
5

BEGIN 和END语句块

awk程序由三部分组成,分别为

初始化:处理输入前做的准备,放在BEGIN块中。
数据处理:处理输入数据。

收尾处理:处理输入完成后要进行的处理,放到END块中。

其中,在“数据处理”过程中,指令被写成一系列模式/动作过程,模式是用于测试输入行的规则,以此确定是否将应用于这些输入行。

awk 的所有代码 都是写在语句块中的

root@itlaoxin41 ~]# awk -F ":" '{print $1}' passwd |head -3
root
ceshi
no such person

语句块可分为 3 类: BEGIN语句块, END语句块,和main语句块

其中 BEGIN 语句块和 END 语句块都是的格式分别为 BEGIN{…} 和 END{…}

main 语句块是一种统称,它的 pattern 部分没有固定格式,也可以省略,main 代码块是在读取文件的每一行的时候都执行的代码块。

且:
BEGIN代码块:

  • 在读取文件之前执行,且执行一次
  • 在BEGIN代码块中,无法使用 $0 或者其他变量

main 代码块:

  • 读取文件时循环执行,(默认情况) 每读取一行,就执行一次 main 代码块
  • main 代码块可有多个

END代码块

  • 在读取文件完成之后执行,且执行一次
  • 有 END 代码块,必有要读取的数据 (可以是标准输入)
  • END 代码块中可以使用 $0 等一些特殊变量,只不过这些特殊变量保存的是最后一轮 awk 循环的数据

常见案例

1. 使用NR行号提取ip

[root@itlaoxin41 ~]# ifconfig |grep inet |awk 'NR==1{print $2}'
192.168.1.41

2. 打印UID小于10的账号名称和UID信息

[root@itlaoxin41 ~]# awk -F : '$3 < 10{print $1,$3}' /etc/passwd
root 0
bin 1
daemon 2
adm 3
lp 4
sync 5
shutdown 6
halt 7
mail 8

3. 数学运算

加法:awk ‘{a=1; b=2; print a+b}’
减法:awk ‘{a=1; b=2; print a-b}’
乘法:awk ‘{a=1; b=2; print a*b}’
除法:awk ‘{a=1; b=2; print a/b}’
取余:awk ‘{a=1; b=2; print a%b}

4. AWK打印硬盘设备名称,默认以空格为分割:

df -h|awk '{print $1}'     $1 第一列

5. AWK以空格、冒号、\t(Tab缩进) 、分号为分割显示第4列内容:

awk -F '[ :\t; ]' '{print $4}' test.txt

6. AWK以冒号分割,打印第一列,同时将内容追加到/tmp/awk.log下:

awk -F: '{print $1 >>"/tmp/awk.log"}' test.txt

7. 打印test.txt文件中的第3行至第5行(2种写法):NR表示打印行,$0表示文本所有域:

awk 'NR==3,NR==5 {print}' test.txt
awk 'NR==3,NR==5 {print $0}' test.txt

8. 打印test.txt文件中的第3行至第5行的第一列与最后一列:

awk 'NR==3,NR==5 {print $1,$NF}' test.txt

备注:默认情况下,awk操作的文本以空格或者缩进格(Tab)为列的分界序列

9. 打印test.txt文件中,长度大于80的行号:

awk 'length($0)>80 {print NR}' test.txt

10. AWK引用Shell变量,使用-v或者双引号+单引号即可:

`awk -v STR=hello '{print STR,$NF}' test.txt`  

 #test.txt每行前加入hello打印
STR="hello";echo|awk '{print "'${STR}'";}'      #需要订正!

总结

AWK没有一万字写不完,这刚刚开始就干到了4500字, 足够你学习用了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/114462.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

UDP用户数据报协议(计算机网络-运输层)

目录 UDP 概述 UDP 的主要特点 UDP 的问题 UDP的多路分用模型 UDP 的首部格式 UDP 概述 用户数据报协议&#xff08;User Datagram Protocol&#xff0c;UDP&#xff09; UDP 只在 IP 的数据报服务之上增加了很少一点的功能&#xff0c;即端口的功能和差错检测的功能 虽…

计算机网络——网络层功能概述

网络层 网络层的主要任务是把分组从源端传到目的端&#xff0c;为分组交换网上的不同主机提供通信服务。网络层的传输单位成为数据报。 数据报是一组比较长的数据&#xff0c;分组则是将数据报划分为不同的片段 网络层的第一个功能&#xff1a;路由的选择和分组的转发。 网络层…

python词云图词频统计

目录 一&#xff1a;安装必要的库 二&#xff1a;数据分析 条形图可视化 三&#xff1a;数据分析 词频统计 词云图可视化 一&#xff1a;安装必要的库 导入必要的库 import collections # 词频统计库 import os import re # 正则表达式库 import urllib.error # 指定url&…

WRF进阶:antro_emiss工具处理全球大气人为排放(EDGRA_HTTPs)/人为排放清单前处理

本内容视频版讲解&#xff1a;全球人为排放处理 介绍 一般人为数据的排放前处理使用pre_chen_src工具&#xff0c;然而pre_chen_src处理后的文件并不是WRF所能读取的文件格式&#xff0c;需要使用onvert_emiss.exe&#xff0c;生成WRF需要的人为排放的nc数据。 在WRF-chem3.6…

煤矿视频监控分析检测 yolo

煤矿视频监控分析检测利用python基于yolo深度学习架构&#xff0c;对现场画面进行实时分析检测。我们使用YOLO(你只看一次)算法进行对象检测。YOLO是一个聪明的卷积神经网络(CNN)&#xff0c;用于实时进行目标检测。该算法将单个神经网络应用于完整的图像&#xff0c;然后将图像…

单片机——LED点阵

1. 基本介绍 LED点阵 LED点阵是由发光二极管排列组成的显示器件&#xff0c;通常应用较多的是88点阵&#xff0c;然后通过多个88点阵组成不同分辨率的LED点阵显示屏&#xff0c;如4个88组成的1616点阵 8*8点阵由64个LED组成&#xff0c;每个LED是放置在行线和列线的交叉点上…

LVGL学习笔记3 - 样式Style

目录 1. 初始化样式 2. 设置样式 3. 添加和移除样式 4. 验证 5. 状态&#xff08;State&#xff09; 6. 部分&#xff08;Parts&#xff09; 样式用于设置对象的外观&#xff0c;比如颜色等属性&#xff0c;存储在 lv_style_t 变量中&#xff0c;这个变量应该是static…

不写一行代码(二):实现安卓基于PWM的LED设备驱动

文章目录一、前言二、系列文章三、准备工作3.1 查找PWM引脚3.2 原理图&#xff1a;确认引脚位置3.3 PWM Controller四、查阅PWM bindings五、编写设备树节点5.1 实现节点&#xff1a;pwm-leds5.2 测试命令六、后语一、前言 在完成了基于GPIO的LED设备驱动的文章后&#xff0c;…

3天学会撰写软件发明专利——3.生命周期

“无意中发现了一个巨牛的人工智能教程&#xff0c;忍不住分享一下给大家。教程不仅是零基础&#xff0c;通俗易懂&#xff0c;而且非常风趣幽默&#xff0c;像看小说一样&#xff01;觉得太牛了&#xff0c;所以分享给大家。点这里可以跳转到教程。”。 一、专利授权生命周期…

4.1 协程:协程基础

1.协程 协程&#xff0c;又称微线程。协程是python个中另外一种实现多任务的方式&#xff0c;只不过比线程更小占用更小执行单元&#xff08;理解为需要的资源&#xff09;。 为啥说它是一个执行单元&#xff0c;因为它自带CPU上下文。这样只要在合适的时机&#xff0c; 我们可…

C++类和对象概念及实现详解(上篇)

文章目录 一、什么是类和对象呢&#xff1f; 1、类的引入 2、类的定义 3、类的访问限定符 4、类对象的储存方式 5、this指针的特性 二、类的六个默认成员函数详解 1、构造函数 2、析构函数 3、未完待续…… 标题&#xff1a;类和对象概念及实现详解&#xff08;上篇&#xff0…

vue3 antd table表格——自定义单元格样式(二)利用rowClassName给table添加行样式

vue3 antd项目实战——修改ant design vue组件中table表格的默认样式&#xff08;二&#xff09;知识调用场景复现修改table表格的行样式一、rowClassName添加行样式二、表格的不可控操作写在最后知识调用 文章中可能会用到的知识链接vue3ant design vuets实战【ant-design-vu…

从头开始用树莓派做一个NAS【最新超详细教程】

一、概述 众所周知在办公的时候两台电脑之间经常倒数据资料非常麻烦&#xff0c;而NAS可以很好的解决这个问题。树莓派搭建NAS方法有很多&#xff0c;我们之前也拍过直接用Samba、FTP这些来实现NAS功能&#xff0c;但是这些需要你会在命令行进行配置&#xff0c;而且对于新手用…

【Linux】Linux权限管理

目录一.Linux用户权限1.权限的概念2.用户分类3.切换用户4.sudo提权二.Linux文件权限1.文件属性2.文件类型3.文件角色划分4.基本权限三.文件访问权限的相关设置方法1.chmod2.chown3.charp4.file5.权限拒绝四.默认权限umask五.目录的权限六.粘滞位1.背景2.准备3.情况4.粘滞位一.L…

初识Docker:(1)什么是docker

初识Docker&#xff1a;&#xff08;1&#xff09;什么是docker项目部署的问题Docker总结项目部署的问题 大型项目组件较多&#xff0c;运行环境也较为复杂&#xff0c;部署时会碰到一些问题&#xff1a; 依赖关系复杂&#xff0c;容易出现兼容性问题开发、测试、生产环境有差…

git revert以及revert的恢复

一&#xff1a;背景与方案 在工作中遇见的这样的场景&#xff1a; 场景一&#xff1a; 已经merge到待发布的版本分支中的功能需要移除当前的分支&#xff0c;改在后续版本发布&#xff0c;示意图如下&#xff0c;展示的是commit序列&#xff0c; 这里想要移除的功能是commi…

[python库] base64库的基本使用

1. base64是什么 base64是一种二进制到文本格式的编码方式。具体来说就是将byte数组编码为字符串的方法&#xff0c;而编码出来的字符串只包含ASCII基础字符。 虽然说base64是一种编码方式&#xff0c;但是它并不推荐作为常规的加密算法使用&#xff0c;因为该算法的加解密算法…

Android开发进阶——binder通讯学习

什么是binder 通常意义下&#xff0c;binder指的是一种通信机制对Server端来说&#xff0c;Binder指的是Binder本地对象&#xff0c;对于Client端来说&#xff0c;Binder指的是Binder代理对象对于传输过程而言&#xff0c;binder是可以跨进程传输的对象 Binder的基本原理 Bi…

【工作流Activiti7】7、Activiti7+SpringBoot

1. 版本问题 1.1. Activiti版本 7.1.0-M6是最后一个支持JDK1.8的版本&#xff0c;此后的版本都要求JDK11以上 目前&#xff0c;Activiti最新版本是7.6.0&#xff0c;它是用JDK11编译的&#xff0c;因此要想使用最新版7.6.0必须升级JDK版本&#xff0c;不能再用1.8 同时&…

【数组中数字出现的次数-有限状态自动机】

数组中数字出现的次数一&#xff0c;有限状态自动机解法二&#xff0c;一般解法想必大家对数组中数字出现的次数的这种题并不少见&#xff0c; 主要有三种&#xff1a; 1&#xff0c;找出数组中只出现一次的数字&#xff08;其他数字出现两次&#xff09; 2&#xff0c;找出数组…