1. 前言
sed全名叫stream editor,流编辑器,用程式的方式来编辑文字,它一次处理一行内容。处理时,把当前处理的行储存在临时缓冲区中,称为“模式空间”(pattern space),接着用sed命令处理模式空间中的内容,处理完成后,把缓冲区的内容送往屏幕。接着处理下一行,这样不断重复,直到档案末尾。档案内容并没有改变,因为这些都在模式空间处理的。Sed可以用来自动编辑一个或多个档案。
Linux批量替换档案内容
普及一下正则表示式的一些基本知识:
^ 表示一行的开头。如:/^#/ 以#开头的匹配。
不能识别此Latex公式:
表示一行的结尾。如:/}
/ 以}结尾的匹配。
> 表示词尾。 如: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
[[email protected] ~]# cp /etc/passwd ~/zcwyou.txt
下文以zcwyou.txt作说明。
4. 使用sed删除输出结果某些行
打印zcwyou.txt的内容并附带行号,不显示3-7行
[[email protected] ~]# nl ~/zcwyou.txt | sed '3,7d'
sed 的动作为 '3,7d' ,d表示删除。同时也要注意的是, sed 后面接的动作,请务必以 '' 两个单引号括住喔!
5. 删除特定的行
把输出结果删除第2行。
[[email protected] ~]# nl ~/zcwyou.txt | sed '2d'
删除带有123的行:
[[email protected] ~]# sed '/123/d' zcwyou.txt
删除第二行
[[email protected] ~]# sed '2d' zcwyou.txt
删除第2行到最后一行。
[[email protected] ~]# sed '2,$d' zcwyou.txt
6. 从某行开始删除
要删除第 3 到最后一行
[[email protected] ~]# nl ~/zcwyou.txt | sed '3,$d'
7. 使用sed新增行
在第二行后面加上字串hello jack
[[email protected] ~]# nl ~/zcwyou.txt | sed '2a hello jack'
在第二行前面加上字串hello jack
[[email protected] ~]# nl ~/zcwyou.txt | sed '2i hello jack'
以行为单位的替换与显示
将第2-5行的内容取代成为Line2-5
[[email protected] ~]# nl ~/zcwyou.txt | sed '2,5c Line2-5'
8. 根据条件仅列出某些行
仅列出 ~/zcwyou.txt 档案内的第 5-7 行
[[email protected] ~]# nl ~/zcwyou.txt | sed -n '5,7p'
9. 使用sed查询资料
打印~/zcwyou.txt的内容,查询带root的行并打印2次。
[[email protected] ~]# nl ~/zcwyou.txt | sed '/root/p'
如果root找到,除了输出所有行,还会输出匹配行。
[[email protected] ~]# nl ~/zcwyou.txt | sed -n '/root/p'
检视关键词并删除
删除~/zcwyou.txt所有包含root的行,输出其他内容
[[email protected] ~]# nl ~/zcwyou.txt | sed '/root/d'
10. 检视关键词并执行命令
搜寻~/zcwyou.txt,找到root对应的行,执行后面花括号中的一组命令,每个命令之间用分号分隔,这里把bash替换为abcshell,再输出这行:
[[email protected] ~]# nl ~/zcwyou.txt | sed -n '/root/{s/bash/abcshell/;p}'
如果只替换~/zcwyou.txt的第一个bash关键字为abcshell,完成后自动退出
[[email protected] ~]# nl ~/zcwyou.txt | sed -n '/bash/{s/bash/abcshell/;p;q}'
11. 打印奇数行或偶数行
方法1:
奇数行
[[email protected] ~]# sed -n 'p;n' zcwyou.txt
偶数行
[[email protected] ~]# sed -n 'n;p' zcwyou.txt
方法2:
奇数行
[[email protected] ~]# sed -n '1~2p' zcwyou.txt
偶数行
[[email protected] ~]# sed -n '2~2p' zcwyou.txt
打印匹配字串的下一行
[[email protected] ~]# grep -A 1 SCC zcwyou.txt
[[email protected] ~]# sed -n '/SCC/{n;p}' zcwyou.txt
[[email protected] ~]# awk '/SCC/{getline; print}' zcwyou.txt
12. 查询关键词并替换
格式:
sed 's/要被取代的字串/新的字串/g'
先看看现有的样版档案内容:
[[email protected] ~]# 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
[Install]
WantedBy=multi-user.target
要求把以上文字中的9001替换为9002
替换命令:
[[email protected] ~]# 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
[Install]
WantedBy=multi-user.target
可以看到,9001已经被换为9002,但这样并没有修改原始档,也没有储存为新档案,我们需要把它重定向储存为新档案。
替换后储存为新档案:
[[email protected] ~]# sed "s/9001/9002/g" sshforwarding.service > sshforwarding9002.service
这样就可以储存为新档案sshforwarding9002.service
如果只替换第4到第8行的文字内容,命令如下:
[[email protected] ~]# sed "4,8s/9001/9002/g" zcwyou.txt
只替换每一行的第一个abc:
[[email protected] ~]# sed 's/abc/S/1' zcwyou.txt
只替换每一行的第二个abc:
[[email protected] ~]# sed 's/abc/S/2' zcwyou.txt
只替换第一行的第3个以后的abc:
[[email protected] ~]# sed 's/abc/S/3g' zcwyou.txt
13. 替换匹配行
使用选项c
把zcwyou.txt的第二行替换为line 2
[[email protected] ~]# sed "2 c line 2" zcwyou.txt
检视带有abc的行,替换内容为line hahaha
[[email protected] ~]# sed "/abc/c line hahaha" zcwyou.txt
sed替换标记
linux sed 替换字串
g 表示行内全面替换。
p 表示打印行。
w 表示把行写入一个档案。
x 表示互换模板块中的文字和缓冲区中的文字。
y 表示把一个字元翻译为另外的字元(但是不用于正则表示式)
子串匹配标记
& 已匹配字串标记
14. sed同时执行多工
一条sed命令,删除~/zcwyou.txt第三行到末尾的资料,并把bash替换为abcshell
[[email protected] ~]# 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
[[email protected] ~]# sed -i 's/a$/b/g' zcwyou.txt
利用 sed 直接在zcwyou.txt 最后一行加入字元end
[[email protected] ~]# sed -i '$a end' zcwyou.txt
$代表的是档案最后一行,而a表示新增内容,因此执行结果是使该档案最后一行新增字元end。不要忘记了,该方法是直接修改原始档,除非你清楚清楚后果,强烈建议使用副本修改并确认无误后才覆盖线上的配置环境。
sed的-i选项非常实用,特别是遇到大档案要修改时,如果你使用vi或vim修改,效率将会非常低。而使用sed,它会自动查询并按要求完成,非常处理速度极为迅速,几乎没有等待。