智传网优云课堂,专注分享IT技术
与IT技术爱好者一起共同学习进步

Linux批量替换文件内容之sed命令详解

1. 前言

sed全名叫stream editor,流编辑器,用程序的方式来编辑文本,它一次处理一行内容。处理时,把当前处理的行存储在临时缓冲区中,称为“模式空间”(pattern space),接着用sed命令处理模式空间中的内容,处理完成后,把缓冲区的内容送往屏幕。接着处理下一行,这样不断重复,直到文件末尾。文件内容并没有改变,因为这些都在模式空间处理的。Sed可以用来自动编辑一个或多个文件。

Linux批量替换文件内容

普及一下正则表达式的一些基本知识:

^ 表示一行的开头。如:/^#/ 以#开头的匹配。
$ 表示一行的结尾。如:/}$/ 以}结尾的匹配。
\< 表示词首。 如:\ 表示词尾。 如:abc> 表示以 abc 結尾的詞。
. 表示任何单个字符。
* 表示某个字符出现了0次或多次。
[ ] 字符集合。 如:[abc] 表示匹配a或b或c,还有 [a-zA-Z] 表示匹配所有的26个字符。如果其中有^表示反,如 [^a] 表示非a的字符

2. sed命令选项

-n :使用安静(silent)模式。在一般 sed 的用法中,所有来自 STDIN 的数据一般都会被列出到终端上。但如果加上 -n 参数后,则只有经过sed 特殊处理的那一行(或者动作)才会被列出来。
-e :直接在命令列模式上进行 sed 的动作编辑;
-f :直接将 sed 的动作写在一个文件内, -f filename 则可以运行 filename 内的 sed 动作;
-r :sed 的动作支持的是延伸型正规表示法的语法。(默认是基础正规表示法语法)
-i :直接修改读取的文件内容,而不是输出到终端。

动作说明: [n1[,n2]]function
n1, n2 :不见得会存在,一般代表『选择进行动作的行数』,举例来说,如果我的动作是需要在 10 到 20 行之间进行的,则『 10,20[动作行为] 』

function:
a\ 在当前行下面插入文本。
i\ 在当前行上面插入文本。
c\ 把选定的行改为新的文本。
d 删除,删除选择的行。
D 删除模板块的第一行。
s 替换指定字符
h 拷贝模板块的内容到内存中的缓冲区。
H 追加模板块的内容到内存中的缓冲区。
g 获得内存缓冲区的内容,并替代当前模板块中的文本。
G 获得内存缓冲区的内容,并追加到当前模板块文本的后面。
l 列表不能打印字符的清单。
n 读取下一个输入行,用下一个命令处理新的行而不是用第一个命令。
N 追加下一个输入行到模板块后面并在二者间嵌入一个新行,改变当前行号码。
p 打印模板块的行。
P(大写) 打印模板块的第一行。
q 退出Sed。
b lable 分支到脚本中带有标记的地方,如果分支不存在则分支到脚本的末尾。
r file 从file中读行。
t label if分支,从最后一行开始,条件一旦满足或者T,t命令,将导致分支到带有标号的命令处,或者到脚本的末尾。
T label 错误分支,从最后一行开始,一旦发生错误或者T,t命令,将导致分支到带有标号的命令处,或者到脚本的末尾。
w file 写并追加模板块到file末尾。
W file 写并追加模板块的第一行到file末尾。
! 表示后面的命令对所有没有被选定的行发生作用。
= 打印当前行号码。
# 把注释扩展到下一个换行符以前。

Linux sed命令详解

3. 测试文件说明

复制/etc/passwd到~/zcwyou.txt

[root@zcwyou ~]# cp /etc/passwd ~/zcwyou.txt

下文以zcwyou.txt作说明。

4. 使用sed删除输出结果某些行

打印zcwyou.txt的内容并附带行号,不显示3-7行

[root@zcwyou ~]# nl ~/zcwyou.txt | sed '3,7d'

sed 的动作为 ‘3,7d’ ,d表示删除。同时也要注意的是, sed 后面接的动作,请务必以 ” 两个单引号括住喔!

5. 删除特定的行

把输出结果删除第2行。

[root@zcwyou ~]# nl ~/zcwyou.txt | sed '2d' 

删除带有123的行:

[root@zcwyou ~]# sed '/123/d' zcwyou.txt

删除第二行

[root@zcwyou ~]# sed '2d' zcwyou.txt

删除第2行到最后一行。

[root@zcwyou ~]# sed '2,$d' zcwyou.txt

6. 从某行开始删除

要删除第 3 到最后一行

[root@zcwyou ~]# nl ~/zcwyou.txt | sed '3,$d' 

7. 使用sed添加行

在第二行后面加上字符串hello jack

[root@zcwyou ~]# nl ~/zcwyou.txt | sed '2a hello jack'

在第二行前面加上字符串hello jack

[root@zcwyou ~]# nl ~/zcwyou.txt | sed '2i hello jack' 

以行为单位的替换与显示

将第2-5行的内容取代成为Line2-5

[root@zcwyou ~]# nl ~/zcwyou.txt | sed '2,5c Line2-5'

8. 根据条件仅列出某些行

仅列出 ~/zcwyou.txt 文件内的第 5-7 行

[root@zcwyou ~]# nl ~/zcwyou.txt | sed -n '5,7p'

9. 使用sed查找数据

打印~/zcwyou.txt的内容,查找带root的行并打印2次。

[root@zcwyou ~]# nl ~/zcwyou.txt | sed '/root/p'

如果root找到,除了输出所有行,还会输出匹配行。

[root@zcwyou ~]# nl ~/zcwyou.txt | sed -n '/root/p'

查看关键词并删除
删除~/zcwyou.txt所有包含root的行,输出其他内容

[root@zcwyou ~]# nl ~/zcwyou.txt | sed  '/root/d'

10. 查看关键词并执行命令

搜索~/zcwyou.txt,找到root对应的行,执行后面花括号中的一组命令,每个命令之间用分号分隔,这里把bash替换为abcshell,再输出这行:

[root@zcwyou ~]# nl ~/zcwyou.txt | sed -n '/root/{s/bash/abcshell/;p}'

如果只替换~/zcwyou.txt的第一个bash关键字为abcshell,完成后自动退出

[root@zcwyou ~]# nl ~/zcwyou.txt | sed -n '/bash/{s/bash/abcshell/;p;q}'    

11. 打印奇数行或偶数行

方法1:
奇数行

[root@zcwyou ~]# sed -n 'p;n' zcwyou.txt

偶数行

[root@zcwyou ~]# sed -n 'n;p' zcwyou.txt

方法2:
奇数行

[root@zcwyou ~]# sed -n '1~2p' zcwyou.txt

偶数行

[root@zcwyou ~]# sed -n '2~2p' zcwyou.txt

打印匹配字符串的下一行

[root@zcwyou ~]# grep -A 1 SCC zcwyou.txt
[root@zcwyou ~]# sed -n '/SCC/{n;p}' zcwyou.txt
[root@zcwyou ~]# awk '/SCC/{getline; print}' zcwyou.txt

12. 查找关键词并替换

格式:
sed ‘s/要被取代的字串/新的字串/g’

先看看现有的样版文件内容:

[root@zcwyou ~]# cat sshforwarding.service 

输出结果:
[Unit]
Description=SSH Forward TCP Port 9001 To SS Server
Documentation=https://linux.die.net/man/1/ssh
After=network.target remote-fs.target nss-lookup.target

[Service]
Type=forking
ExecStart=/usr/bin/ssh -NT -f -L my_public_ip:9001:SS_server:9001 localhost
ExecReload=/bin/kill -HUP $MAINPID
ExecStop=/bin/kill -HUP $MAINPID

[Install]
WantedBy=multi-user.target

要求把以上文本中的9001替换为9002
替换命令:

[root@zcwyou ~]# sed "s/9001/9002/g" sshforwarding.service

输出结果:
[Unit]
Description=SSH Forward TCP Port 9002 To Shadowsocks Server
Documentation=https://linux.die.net/man/1/ssh
After=network.target remote-fs.target nss-lookup.target

[Service]
Type=forking
ExecStart=/usr/bin/ssh -NT -f -L my_public_ip:9002:shadowsocks_server:9002 localhost
ExecReload=/bin/kill -HUP $MAINPID
ExecStop=/bin/kill -HUP $MAINPID

[Install]
WantedBy=multi-user.target

可以看到,9001已经被换为9002,但这样并没有修改源文件,也没有保存为新文件,我们需要把它重定向保存为新文件。

替换后保存为新文件:

[root@zcwyou ~]# sed "s/9001/9002/g" sshforwarding.service > sshforwarding9002.service

这样就可以保存为新文件sshforwarding9002.service

如果只替换第4到第8行的文本内容,命令如下:

[root@zcwyou ~]# sed "4,8s/9001/9002/g" zcwyou.txt

只替换每一行的第一个abc:

[root@zcwyou ~]# sed 's/abc/S/1' zcwyou.txt

只替换每一行的第二个abc:

[root@zcwyou ~]# sed 's/abc/S/2' zcwyou.txt

只替换第一行的第3个以后的abc:

[root@zcwyou ~]# sed 's/abc/S/3g' zcwyou.txt

13. 替换匹配行

使用选项c
把zcwyou.txt的第二行替换为line 2

[root@zcwyou ~]# sed "2 c line 2" zcwyou.txt

查看带有abc的行,替换内容为line hahaha

[root@zcwyou ~]# sed "/abc/c line hahaha" zcwyou.txt

sed替换标记

linux sed 替换字符串

g 表示行内全面替换。
p 表示打印行。
w 表示把行写入一个文件。
x 表示互换模板块中的文本和缓冲区中的文本。
y 表示把一个字符翻译为另外的字符(但是不用于正则表达式)
\1 子串匹配标记
& 已匹配字符串标记

14. sed同时执行多任务

一条sed命令,删除~/zcwyou.txt第三行到末尾的数据,并把bash替换为abcshell

[root@zcwyou ~]# nl ~/zcwyou.txt | sed -e '3,$d' -e 's/bash/abcshell/'

-e表示多点编辑,第一个动作是删除~/zcwyou.txt第三行到末尾的数据,第二条命令搜索bash替换为abcshell。

15. 使用sed直接修改文件内容(危险不建议)

sed 可以直接修改文件的内容,不必使用管道命令或数据流重导向! 不过,由於这个动作会直接修改到原始的文件,所以建议找一个无用的文件做测试,比如本文里的zcwyou.txt,仅仅用来实验研究。

利用sed将zcwyou.txt内的每一行结尾a换成b

[root@zcwyou ~]# sed -i 's/a$/b/g' zcwyou.txt

利用 sed 直接在zcwyou.txt 最后一行加入字符end

[root@zcwyou ~]# sed -i '$a end' zcwyou.txt

$代表的是文件最后一行,而a表示新增内容,因此执行结果是使该文件最后一行新增字符end。不要忘记了,该方法是直接修改源文件,除非你清楚清楚后果,强烈建议使用副本修改并确认无误后才覆盖线上的配置环境。

sed的-i选项非常实用,特别是遇到大文件要修改时,如果你使用vi或vim修改,效率将会非常低。而使用sed,它会自动查找并按要求完成,非常处理速度极为迅速,几乎没有等待。

赞(0)
未经允许不得转载:Linux入门学习到精通 » Linux批量替换文件内容之sed命令详解
分享到: 更多 (0)

学习QQ群:557371664

关注微信公众号自助视频学习

评论 抢沙发

评论前必须登录!