Sed简介
Sed是Stream Editor(流编辑器)缩写,是操作、过滤和转换文本内容的强大工具,常用功能有增删改查。
Sed命令执行流程
Sed语法格式
Sed [option] ‘[匹配][处理]’ [file]
说明:个人将语法中sed命令部分分为先匹配后处理两个部分。Sed可以接文件,也可以接标准输入,比如管道等。
查看Sed版本
[root@web01 mnt]# sed --version
GNU sed version 4.2.1
统一实验文本
[root@web01 sed]# cat >> person.txt << EOF
> 101,peterwang,CEO
> 102,zhangyao,CTO
> 103,Alex,COO
> 104,yy,CFO
> 105,feixue,CIO
> EOF
单行增加
[root@web01 sed]# sed '2a 106,dandan,CSO' person.txt
101,peterwang,CEO
102,zhangyao,CTO
106,dandan,CSO // a 追加文本到指定行后
103,Alex,COO
104,yy,CFO
105,feixue,CIO [root@web01 sed]# sed '2i 106,dandan,CSO' person.txt
101,peterwang,CEO
106,dandan,CSO // i 插入文本到指定行前
102,zhangyao,CTO
103,Alex,COO
104,yy,CFO
105,feixue,CIO
多行增加
[root@web01 sed]# sed '2a 106,dandan,CSO\n107,bingbing,CCO' person.txt
101,peterwang,CEO
102,zhangyao,CTO
106,dandan,CSO
107,bingbing,CCO
103,Alex,COO
104,yy,CFO
105,feixue,CIO [root@web01 sed]# sed '2a 106,dandan,CSO\
> 107,bingbing,CCO' person.txt
101,peterwang,CEO
102,zhangyao,CTO
106,dandan,CSO
107,bingbing,CCO
103,Alex,COO
104,yy,CFO
105,feixue,CIO
匹配:指定执行的行范围
Sed可以对特定的行进行处理,如果不指定那么sed默认匹配所有行。
用法:n1[,n2]{sed-commands} 地址用逗号分割,可以是数字、正则或二者的组合。
举例:
- 10{sed-commands} 对第10行操作
- 10,20{sed-commands} 对10-20行进行操作,包括第10,20行
- 10,+20{sed-commands} 对10-30行进行操作(从第十行开始向后20行),包括第10,30行
- 1~2{sed-commands} 对 1,3,5,7…行进行操作(~2表示间隔为2)
- 10,${sed-commands} 对10到最后一行进行操作($代表最后一行),包括第10行
- /peter/{sed-commands} 匹配peter所在的行进行操作
- /peter/,/Alex/{sed-commands} 对peter所在行到Alex所在行进行操作
- /peter/,10{sed-commands} 对peter所在行到第10行进行操作
- 1,/Alex/{sed-commands} 对第1行到Alex所在行进行操作
- /peter/,+2{sed-commands} 对peter所在行到其后2行进行操作
删(d)
[root@web01 sed]# sed '2d' person.txt
101,peterwang,CEO
103,Alex,COO
104,yy,CFO
105,feixue,CIO [root@web01 sed]# sed '2,5d' person.txt
101,peterwang,CEO [root@web01 sed]# sed '2,$d' person.txt
101,peterwang,CEO [root@web01 sed]# sed '1~2d' person.txt
102,zhangyao,CTO
104,yy,CFO [root@web01 sed]# sed '1,+2d' person.txt
104,yy,CFO
105,feixue,CIO [root@web01 sed]# sed '/peter/d' person.txt
102,zhangyao,CTO
103,Alex,COO
104,yy,CFO
105,feixue,CIO [root@web01 sed]# sed '/peter/,/Alex/d' person.txt
104,yy,CFO
105,feixue,CIO [root@web01 sed]# sed '/peter/,3d' person.txt
104,yy,CFO
105,feixue,CIO
整行替换(c)
[root@web01 sed]# sed '2c 106,dandan,CSO' person.txt
101,peterwang,CEO
106,dandan,CSO
103,Alex,COO
104,yy,CFO
105,feixue,CIO
部分替换(s)
基本格式: sed ‘s#匹配文本#替换文本#g’ sed –i ‘s#匹配文本#替换文本#g’
- s 是sed命令,表示替换;
- g 是命令s的替换标记,表示对该行全局替换,不加g则只替换每一行匹配到的第一个文本;
- # 是定界符,可以是/ # @等任意字符,习惯上用 #;
- 匹配文本可以用正则表达式,替换文本不能。
- - i 是sed的一个选项,表示对文件进行修改,通常不加i时,sed只是对内存模式空间里的数据进行操作,不修改磁盘中的源文件。
[root@web01 sed]# sed 's#zhangyao#peter#g' person.txt
101,peterwang,CEO
102,peter,CTO
103,Alex,COO
104,yy,CFO
105,feixue,CIO [root@web01 sed]# sed -i 's#zhangyao#peter#g' person.txt
[root@web01 sed]# cat person.txt
101,peterwang,CEO
102,peter,CTO
103,Alex,COO
104,yy,CFO
105,feixue,CIO
变量替换
#支持使用变量表示数据
[root@web01 sed]# cat >> test.txt << EOF
> a
> b
> a
> EOF
[root@web01 sed]# x=a
[root@web01 sed]# y=b
[root@web01 sed]# echo $x $y
a b
[root@web01 sed]# sed 's#'$x'#'$y'#g' test.txt
b
b
b
分组替换
基本格式
#分组替换常用于取某一行里的动态数据
[root@web01 sed]# echo "I am Linux student" | sed -r 's#^.*am (.*) stu.*$#\1#g'
Linux // \1 表示的是前面()里的内容 -r 表示使用扩展正则
案例:取IP地址
[root@web01 sed]# ifconfig eth0 // 取出10.0.0.8
eth0 Link encap:Ethernet HWaddr 00:0C:29:1D:68:4B
inet addr:10.0.0.8 Bcast:10.0.0.255 Mask:255.255.255.0
inet6 addr: fe80::20c:29ff:fe1d:684b/64 Scope:Link
UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1
RX packets:10434 errors:0 dropped:0 overruns:0 frame:0
TX packets:7185 errors:0 dropped:0 overruns:0 carrier:0
collisions:0 txqueuelen:1000
RX bytes:4253413 (4.0 MiB) TX bytes:742419 (725.0 KiB) [root@web01 sed]# ifconfig eth0 | sed -nr 's#^.*dr:(.*) Bc.*$#\1#gp'
10.0.0.8
[root@web01 sed]# ifconfig eth0 | awk -F "[ :]+" 'NR==2{print $4}' // 推荐
10.0.0.8
[root@web01 sed]# grep "IPADDR" /etc/sysconfig/network-scripts/ifcfg-eth0 | cut -d = -f 2
10.0.0.8
按行查询
[root@web01 sed]# sed -n '2p' person.txt // -n 是sed 选项,表示取消默认输出
102,zhangyao,CTO [root@web01 sed]# sed -n '2,3p' person.txt
102,zhangyao,CTO
103,Alex,COO
按字符串查询
[root@web01 sed]# sed -n '/CTO/p' person.txt
102,zhangyao,CTO [root@web01 sed]# sed -n '/CTO/,/CFO/p' person.txt
102,zhangyao,CTO
103,Alex,COO
104,yy,CFO
混合查询
[root@web01 sed]# sed -n '2,/CFO/p' person.txt
102,zhangyao,CTO
103,Alex,COO
104,yy,CFO
特殊用法
-i选项修改前备份,-i后面可以接后缀,这样sed就会先备份再修改
[root@web01 sed]# cat person.txt
101,peterwang,CEO
102,zhangyao,CTO
103,Alex,COO
104,yy,CFO
105,feixue,CIO
[root@web01 sed]# sed -i.bak '2s#zhangyao# #g' person.txt
[root@web01 sed]# ls
person.txt person.txt.bak test.txt
[root@web01 sed]# head -2 person.txt
101,peterwang,CEO
102, ,CTO
[root@web01 sed]# head -2 person.txt.bak
101,peterwang,CEO
102,zhangyao,CTO
只对匹配到的第一行进行操作
比如我们要修改nginx端口
#第一种方法会匹配所有包含listen的行然后将80替换为8080
sed -i '/listen/{s/80/8080/}' nginx.conf.default
#第二种方法指定了一个范围,替换范围种所有的80为8080,但如果我们确定该范围内只有listen关键字行包含80,实际上这就相当于只对匹配到的第一个listen行进行端口替换
sed -i '0,/listen/{s/80/8080/}' nginx.conf.default