sed - stream editor for filtering and transforming text
Sed是一个流编辑器。流编辑器用于对输入流(文件或管道输入)执行基本的文本转换。虽然在某些方面类似于允许脚本编辑的编辑器(如ed),但sed的工作方式是只对输入进行一次传递,因此效率更高。但是sed能够过滤管道中的文本,这一点与其他类型的编辑器有很大的区别。
格式:一次处理一行内容,在处理时,把当前处理的行存储在临时缓冲区中当中,该缓冲区称为模式空间(Pattern space),接着用sed命令处理缓冲区中的内容,处理完毕后,把缓冲区的内容送到标准输出;然后紧接着去处理下一行,重复完成相同的操作,直至文件未尾;sed处理的整个过程中,对象文件中的内容并没有改变,除非使用重定向来存储处理后的结果,sed主要用来自动编辑一个或多个文件,简化对文件的反复操作过程。默认不编辑原文件,仅对模式空间中的数据进行处理
sed [option] ... '[address-range][script]' inputfile...
常用选项:
- -n:不输出模式中的内容至屏幕
- -e:多点编辑
- -r:支持扩展正则表达式
- -f /path/to/sript_file:从指定文件中读取编辑脚本
- -i:原处编辑
- 不给定地址,默认将对全文进行处理
- 单地址:
- #:指定的行
- /pattern/:被此模式匹配的每一行
- $:最后一行,$-1就是倒数第二行
- 地址范围:
- #,#:1,3第一行到第三行
- #,+#:1,+2第一行到1+2=3行
- /pattern1/,/pattern2/:从第一个匹配到pattern1到第一个匹配pattern2的中间所有行
- #,/pattern/:从第多少行到第一个匹配pattern的行中间所有行
pattern
支持正则表达式方法。
- d:删除
- p:显示模式空间的内容
- c \test:将选定的行替换为文本
- a \test:在行后面追加文本,支持使用\n实现多行追加
- i \test:在行前面追加文本,支持使用\n实现多行追加
- r /path/to/somefile:读取指定文件的文本流至模式空间中匹配的行的行后
- w /path/to/somefile:保存模式空间匹配到的行至指定文件中
- =:为模式空间中行打印行号
- !:取反条件,写于位置后
- s/// s@pattern@replace@ s### //如果pattern和replace中有的字符与分隔符(s后的和一个字符)相同,即可用\转义,也可将修改成不同的分隔符(如!@#$%^&*),这样阅读起来方便些,避免不必要的错误。 替换标记: - g:行内全局替换
s@@@g
。不加g默认只替换每行中第一次被匹配到的串 - i:忽略字符大小写s@@@i
注:这里的a
,c
,i
,r
,w
参数都是另起一行追加,并不是在原有行首或行尾追加。
[root@node1 tmp]# echo first | sed '1p' #sed默认是将模式空间的内容输出至屏幕,同时又用p选项输出,所以就会输出两次firstfirst[root@node1 tmp]# echo first | sed -n '1p' #使用-n选项就只输出想要的行,而不是把模式空间中的内容输出first[root@node1 tmp]# echo -e "222\n222"| sed 's/2/1/' #懒惰122122[root@node1 tmp]# echo -e "222\n222"| sed 's/2/1/g' #如果编辑命令后不加g,默认只修改每一行中第一个匹配的值,加上g则表示全局修改(勤奋)111111[root@node1 tmp]# cat a.sh1end12end23end34end4[root@node1 tmp]# sed -n '/1/,/end/p' a.sh1end1[root@node1 tmp]# sed -n '1,/end/p' a.sh #位置匹配是懒惰的,只匹配第一次匹配到的位置 1end1[root@node1 tmp]# echo -e "222"| sed 's/2/[&]/g' #可以使用&符号代替匹配到的内容,然后在其前后增加中括号 [2][2][2][root@node1 tmp]# echo -e "222"| sed -e 's/2/1/g' -e 's/1/3/' #多个匹配311
sed通常用单引号来引用;也可使用双引号,使用双引号后,双引号会对表达式求值:
$ name=walter$ echo 'my name is $name' | sed "s@\$name@$name@"my name is walter1.删除grub的启动项文件中所有以空白开头的行行首的空白字符
sed 's/^[[:space:]]*//' /etc/grub2.cfgsed 's/[[:space:]]\+//' /etc/grub2.cfg
2.echo一个绝对路径给sed命令,取出其基名,取出其目录名。
echo /etc/sysconfig/network-scripts/ | sed -r 's@(/.*/)[^/]+/?@\1@' 目录名echo /etc/sysconfig/network-scripts/ | sed -r 's@/.*/([^/]+)/?@\1@' 基名
3.删除某一行
sed -i '/xxx/d' a.sh
4.注释10-20
sed -i '10,20s@^@#@g' xxx.sh
关键知识:
圆括号括起来的正则表达式所匹配的字符串会可以当成变量来使用,使用方式\1,\2..,按照括号从左到右的顺序给定变量名称[root@node1 tmp]# echo -e "121"| sed -r 's@(12)@\13@g' #在匹配的后面加值1231[root@node1 tmp]# echo -e "121"| sed -r 's@1(12)@\13@g' #这个是匹配失败,返回原来的值,匹配不到112121[root@node1 tmp]# echo "1*2 2*1" | sed -r 's@(\w*\w)\s?(\w\*\w)@\1=\2@' #\1是1*2,\2是2*11*2=2*1$ echo 'this is en example' | sed -E 's@(\w+)@[\1]@g'[this] [is] [en] [example]总结:
sed
命令按行编辑,拥有pattern space和hold space两种模式空间,日常所用也就pattern space,但是hold space个人觉得就像是汉诺塔一样有第三个柱子,就多出了很多种玩法了。pattern space就相当于只有两个柱子,玩法自然就少了。vim
文本编辑器里的替换功能和sed
命令语法一致。所以需要熟练掌握。 hold space高级玩法资料: