IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 系统运维 -> RE、grep、sed与awk -> 正文阅读

[系统运维]RE、grep、sed与awk

课程来源:https://www.bilibili.com/video/BV1Kg411g7bC

grep用于在文件中查找:

daniel@ubuntu:~/sh$ grep 'ui' id.txt 
oldboy 57089234758ui

正则表达式

^开头

^x查询以x开头的行:grep '^my' oldboy.txt

daniel@ubuntu:~/sh$ cat -n oldboy.txt 
     1	I am oldboy teacher!
     2	I teach linux
     3	
     4	I like badminton ball ,billiard ball and chinese chess!
     5	my blog is http://oblboy.blog.51cto.com
     6	our size is http://blog.oldboyedu.com
     7	my qq is 49000448
     8	
     9	not 4900000448
    10	my god ,i am not oldbey,but OLDBOY!
daniel@ubuntu:~/sh$ grep -n '^my' oldboy.txt
5:my blog is http://oblboy.blog.51cto.com
7:my qq is 49000448
10:my god ,i am not oldbey,but OLDBOY!

$结尾

x$查询以x结尾的行:grep '448$' oldboy.txt

daniel@ubuntu:~/sh$ grep '448$' oldboy.txt
my qq is 49000448
not 4900000448

cat -A显示文件中的空行和空格:

daniel@ubuntu:~/sh$ cat -A oldboy.txt 
I am oldboy teacher!$
I teach linux$
$
I like badminton ball ,billiard ball and chinese chess!$
my blog is http://oblboy.blog.51cto.com $
our size is http://blog.oldboyedu.com $
my qq is 49000448$
$
not 4900000448$
my god ,i am not oldbey,but OLDBOY!$

^$空行

^$

daniel@ubuntu:~/sh$ grep -n '^$' oldboy.txt 
3:
8:

排除空行:grep -v '^$' oldboy.txt,-v 或 --invert-match : 显示不包含匹配文本的所有行。

.任意一个字符

但是不匹配空行

daniel@ubuntu:~/sh$ grep '.' oldboy.txt 
I am oldboy teacher!
I teach linux
I like badminton ball ,billiard ball and chinese chess!
my blog is http://oblboy.blog.51cto.com 
our size is http://blog.oldboyedu.com 
my qq is 49000448
not 4900000448
my god ,i am not oldbey,but OLDBOY!

\转义字符

daniel@ubuntu:~/sh$ grep '\.$' oldboy.txt 
I teach linux.

*前一个字符连续出现0次及以上

grep '0*' oldboy.txt ,数字0出现0次及以上

.*表示所有

grep '.*' oldboy.txt

对于每一行查找到字母t:grep '^.*t' oldboy.txt

正则贪婪性,会一直匹配到最后一个字符:
在这里插入图片描述

[abc]匹配abc中的任何一个

grep '[abc]' oldboy.txt

匹配小写字母:grep '[a-z]' oldboy.txt
匹配大写字母:grep '[A-Z]' oldboy.txt
匹配数字:grep '[0-9]' oldboy.txt

匹配所有大小写字母和数字:

grep '[a-zA-Z0-9]' oldboy.txt
grep '[a-Z0-9]' oldboy.txt
grep -i '[a-z0-9]' oldboy.txt	#-i表示忽略大小写

注意这里的“或”逻辑是直接将模式串拼在一起,不需要用|显示的指出。事实上,在[]中的模式串中的字符没有特殊含义,仅表示该字符本身。

grep '[a-z|A-Z|0-9]' oldboy.txt		#会连同'|'一起匹配
grep '[a-z,A-Z,0-9]' oldboy.txt		#会连同','一起匹配
grep '[a-z A-Z 0-9]' oldboy.txt		#会连通' '一起匹配

[^abc]排除abc

grep '[^abc]' oldboy.txt

不以mn开头的行:grep '^[^mn]' oldboy.txt

不以数字结尾的行:grep '[^0-9]$' oldboy.txt

基础正则总结

在这里插入图片描述

+前一个字符出现1次及以上

首先有3中方式使用扩展正则:

egrep '[0+]' oldboy.txt		#使用egrep命令
grep -E '[0+]' oldboy.txt	#-E表示扩展正则
grep '[0\+]' oldboy.txt		#转义

匹配文件中所有的数字或单词:

grep '[0-9]+' oldboy.txt
grep -o '[0-9]+' oldboy.txt		#只显示被匹配的部分
grep '[a-z]' oldboy.txt
grep -o '[a-z]' oldboy.txt		#只显示被匹配的部分

|或者

egrep 'oldboy|oldbey' oldboy.txt
[]的区别:[]一次只匹配一个字符,而|用于匹配单词

()表示一个字符整体

egrep 'oldb(o|e)y' oldboy.txt

o{n,m}前一个字符出现n到m次

egrep '0{3,4}' oldboy.txt,表示0至少出现3次,至多出现4次
egrep '0{3}' oldboy.txt,表示0正好出现3次
egrep '0{3,}' oldboy.txt,表示0至少出现3次
egrep '0{,3}' oldboy.txt,表示0正好至多3次

?前一个字符出现0或1次

egrep 'go?d' oldboy.txt

扩展正则总结

在这里插入图片描述
匹配身份证号:

egrep '[0-9]{17}[0-9X]' id.txt

排除空行或者含有#的行

egrep -v '^$|#' file.txt

总结:
在这里插入图片描述
三剑客:
P16

grep

参数含义
-Aafter
-Bbefore
-Ccontex
-ccount
-vinvert
-nline number
-iignore
-wword-regexp精确匹配
daniel@ubuntu:~/sh$ seq 10 | grep -A5 2
2
3
4
5
6
7
daniel@ubuntu:~/sh$ seq 10 | grep -B3 9
6
7
8
9
daniel@ubuntu:~/sh$ seq 10 | grep -C4 3
1
2
3
4
5
6
7

-c统计行数

daniel@ubuntu:~/sh$ ps -ef | grep sshd | wc -l
2
daniel@ubuntu:~/sh$ ps -ef | grep -c sshd
2

-v可以排除,可以用于排除ps结果中的grep进程:ps -ef | grep cround | grep -v grep

-w用于精确匹配

daniel@ubuntu:~/sh$ grep -w 22 /etc/services
ssh		22/tcp				# SSH Remote Login Protocol

sed

stream editor

查找p

-n:仅显示script处理后的结果

格式含义实例
‘3p’第3行sed -n ‘3p’ oldboy.txt
‘4,7p’第4到7行sed -n ‘4,7p’ oldboy.txt
‘4,$p’第4到末尾sed -n ‘4,$p’ oldboy.txt
‘$p’到末尾sed -n ‘$p’ oldboy.txt
‘/[RE]/p’/regular expression/正则匹配sed -n ‘/[45]/p’ oldboy.txt
sed -n ‘/oldboy/p’ oldboy.txt
‘/102/,/105/p’从包含102的行到包含105的行,表示范围sed -n ‘/11:02:00/,/11:03:01/p’ oldboy.txt

删除d

-r:支持扩展正则;查找的过程同上
删除空行或包含#的行:sed -r '/^$|#/d' oldboy.txt或者用!取反的psed -nr '/^$|#/!p' oldboy.txt

增加cai

在第3行的下一行添加:sed '3a 996daniel,hello' oldboy.txt
在第3行的上一行添加:sed '3i 996daniel,hello' oldboy.txt
替换掉第3行:sed '3c 996daniel,hello' oldboy.txt
在结尾添加:sed '$a hello,daniel' oldboy.txt

替换s

sed 's#a#b#g' file.txt
s表示替换
a表示要被替换的内容
b表示目标内容
g表示全局,如果不加,只匹配每一行第一个内容

sed 's#[0-9]##g' oldboy.txt
sed 's#[0-9]#*#g' oldboy.txt

后向引用和反向引用

先保护起来再使用
在123456左右加上<>:

echo 123456 | sed -r 's#(.*)#<\1>#'

交换前后位置:sed 's#^(.*)_(.*)$#\2_\1#'

daniel@ubuntu:~/sh$ echo fuck_you | sed -r 's#^(.*)_(.*)$#\2_\1#'
you_fuck
#or
daniel@ubuntu:~/sh$ echo fuck_you | sed -r 's#^([a-z]+)_([a-z]+)$#\2_\1#'
you_fuck

显示网卡的ip信息:

daniel@ubuntu:~/sh$ ip a s ens33
2: ens33: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc fq_codel state UP group default qlen 1000
    link/ether 00:0c:29:56:dc:a5 brd ff:ff:ff:ff:ff:ff
    inet 192.168.37.139/24 brd 192.168.37.255 scope global dynamic noprefixroute ens33
       valid_lft 1632sec preferred_lft 1632sec
    inet6 fe80::a362:1f1f:25bf:bd2f/64 scope link noprefixroute 
       valid_lft forever preferred_lft forever

daniel@ubuntu:~/sh$ ip a s ens33 | sed -n '3p' | sed -r 's#^.*inet (.*) brd.*$#\1#'
192.168.37.139/24
daniel@ubuntu:~/sh$ ip a s ens33 | sed -rn '3 s#^.*inet (.*) brd.*$#\1# p'
192.168.37.139/24

awk

执行过程

awk 'BEGIN{print 1/3}'

在这里插入图片描述

按列拆分:awk -F- 'BEGIN{print "ip"}{print $1}END{print "end of file"}' log.txt

daniel@ubuntu:~/sh$ cat log.txt 
192.168.1.20 - - [21/Apr/2020:14:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.21 - - [21/Apr/2020:15:27:49 +0800] "GET /2/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.22 - - [21/Apr/2020:21:27:49 +0800] "GET /3/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.23 - - [21/Apr/2020:22:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.24 - - [22/Apr/2020:15:27:49 +0800] "GET /2/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.25 - - [22/Apr/2020:15:26:49 +0800] "GET /3/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.20 - - [23/Apr/2020:08:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.21 - - [23/Apr/2020:09:20:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.22 - - [23/Apr/2020:10:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.22 - - [23/Apr/2020:10:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.20 - - [23/Apr/2020:14:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.21 - - [23/Apr/2020:15:27:49 +0800] "GET /2/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.22 - - [23/Apr/2020:15:27:49 +0800] "GET /3/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.25 - - [23/Apr/2020:16:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.24 - - [23/Apr/2020:20:27:49 +0800] "GET /2/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.25 - - [23/Apr/2020:20:27:49 +0800] "GET /3/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.20 - - [23/Apr/2020:20:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.21 - - [23/Apr/2020:20:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.22 - - [23/Apr/2020:20:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.22 - - [23/Apr/2020:22:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
192.168.1.21 - - [23/Apr/2020:23:27:49 +0800] "GET /1/index.php HTTP/1.1" 404 490 "-" "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:45.0) Gecko/20100101 Firefox/45.0"
daniel@ubuntu:~/sh$ awk -F- 'BEGIN{print "ip"}{print $1}END{print "end of file"}' log.txt 
ip
192.168.1.20 
192.168.1.21 
192.168.1.22 
192.168.1.23 
192.168.1.24 
192.168.1.25 
192.168.1.20 
192.168.1.21 
192.168.1.22 
192.168.1.22 
192.168.1.20 
192.168.1.21 
192.168.1.22 
192.168.1.25 
192.168.1.24 
192.168.1.25 
192.168.1.20 
192.168.1.21 
192.168.1.22 
192.168.1.22 
192.168.1.21 
end of file

但是awk默认是按空格拆分的:awk 'BEGIN{print "ip"}{print $1}END{print "end of file"}' log.txt

明确概念:
行=记录=record
列=域=字段=filed

取行

NR: Number of Record
NR==1:第1行
NR>=1 && NR<=5:第1-5行
/oldboy/:包含’oldboy’的行
/102/,/105/:包含102到105的行

awk 'NR==1' oldboy.txt
awk 'NR>=1 && NR<=5' oldboy.txt
awk '/oldboy/' oldboy.txt

取列

-F分隔符:默认空格,多个空格或tab键
$n:取出第n列,ls -l | awk '{print $5}'
$0:表示整行的内容,ls -l | awk 'NR==3{print $0}'
column会按列对齐:ls -l | awk '{print $7,$9}'| column -t

awk的内置变量含义
NR行号
NF每行有多少个字段(列数),则$NF表示最后一列
FSFiled Separator,字段分隔符
OFSOutput Filed Separator,输出字段分隔符
awk -F: '{print $1" add_something "$NF}' /etc/passwd | column -t

令输出分隔符为,

awk -F: -vOFS=, '{print $NF,$2,$1}' info.txt

取出网卡的ip地址:ip a s ens33 | awk 'NR==3' | awk -F"[ /]+" '{print $3}'

daniel@ubuntu:~/sh$ ip a s ens33
2: ens33: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc fq_codel state UP group default qlen 1000
    link/ether 00:0c:29:56:dc:a5 brd ff:ff:ff:ff:ff:ff
    inet 192.168.37.139/24 brd 192.168.37.255 scope global dynamic noprefixroute ens33
       valid_lft 1416sec preferred_lft 1416sec
    inet6 fe80::a362:1f1f:25bf:bd2f/64 scope link noprefixroute 
       valid_lft forever preferred_lft forever
daniel@ubuntu:~/sh$ ip a s ens33 | awk 'NR==3' | awk -F"[ /]+" '{print $3}'
192.168.37.139

或者:ip a s ens33 | awk -F"[ /]+" 'NR==3 {print $3}'

模式匹配

匹配第3列中以1开头的行

awk -F: '$3~/^1/' /etc/passwd

为条件加上动作

awk -F: '$3~/^2/{print $NF,$2,$3}' /etc/passwd

awk -F: '$3~/^[12]/{print $1,$3,$NF}' /etc/passwd
awk -F: '$3~/^1|^2/{print $1,$3,$NF}' /etc/passwd | column -t
awk -F: '$3~/^(1|2)/{print $1,$3,$NF}' /etc/passwd | column -t

范围匹配:

/begin/,/end/
NR==3,NR==5

查询日志中的ip:

awk '/06:16:.* /,/06:16:.* / {print $1,$7}' access.log

BEGIN和END
在这里插入图片描述
求从1加到100

seq 100 | awk '{sum=sum+$1}END{print sum}'

数组

一般用于统计日志,如每个ip的出现次数或每个状态码出现的次数

awk 'BEGIN{a[0]="hello";a[1]="world";print a[0],a[1]}'

输出数组的全部内容:

#shell的做法
for i in ${array[*]}
do
	echo $i
done

#awk的做法
awk 'BEGIN{a[0]="hello";a[1]="world";for (i in a) print i,a[i]}'

统计次数:awk -F"[/.]+" '{cnt[$2]++}END{for(i in cnt) print i,cnt[i]}' url.txt

daniel@ubuntu:~/sh$ cat url.txt 
http://www.etiantian.org/index.html
http://www.etiantian.org/1.html
http://post.etiantian.org/index.html
http://mp3.etiantian.org/index.html
http://www.etiantian.org/3.html
http://post.etiantian.org/2.html
daniel@ubuntu:~/sh$ awk -F"[/.]+" '{cnt[$2]++}END{for(i in cnt) print i,cnt[i]}' url.txt 
www 3
post 2
mp3 1

根据第2列的数字逆向排序:

daniel@ubuntu:~/sh$ awk -F"[/.]+" '{cnt[$2]++}END{for(i in cnt) print i,cnt[i]}' url.txt | sort -rnk2
www 3
post 2
mp3 1

控制流

for循环

#shell中的for循环
for((i=0;i<10;++i))
do
	echo $i
done

#awk中的for循环
awk 'BEGIN{for(i=0;i<10;++i) print i}'
awk 'BEGIN{for(i=1;i<=100;++i) sum+=i;print sum}'
#print也参与循环
awk 'BEGIN{for(i=1;i<=100;++i) {sum+=i;print sum}}'

if

#shell中的if
if [ age -eq 18 ]
then
	echo "hello,man"
fi
#awk中的if
seq 100 | awk '{if($0%5==0) print "yes";else print "no"}'
awk 'BEGIN{for(i=0;i<100;++i){ if(i%5==0) print "yes";else print "no";}}'

例题:统计字符数小于6的单词:

echo I am oldboy teacher welcome to oldboy training class
awk -F"[ .]" '{for(i=1;i<=NF;++i) if(length($i)<6) print $i;}'
  系统运维 最新文章
配置小型公司网络WLAN基本业务(AC通过三层
如何在交付运维过程中建立风险底线意识,提
快速传输大文件,怎么通过网络传大文件给对
从游戏服务端角度分析移动同步(状态同步)
MySQL使用MyCat实现分库分表
如何用DWDM射频光纤技术实现200公里外的站点
国内顺畅下载k8s.gcr.io的镜像
自动化测试appium
ctfshow ssrf
Linux操作系统学习之实用指令(Centos7/8均
上一篇文章      下一篇文章      查看所有文章
加:2022-04-30 09:04:54  更:2022-04-30 09:07:24 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/6 19:33:08-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码