sed

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:原处编辑

**地址定界(address-range):** 1. 不给定地址,默认将对全文进行处理 2. 单地址: * #:指定的行 * /pattern/:被此模式匹配的每一行 * $:最后一行,$-1就是倒数第二行 3. 地址范围: * #,#:1,3第一行到第三行 * #,+#:1,+2第一行到1+2=3行 * /pattern1/,/pattern2/:从第一个匹配到pattern1到第一个匹配pattern2的中间所有行 * #,/pattern/:从第多少行到第一个匹配pattern的行中间所有行

pattern支持正则表达式方法。


**编辑命令:** * d:删除 * p:显示模式空间的内容 * c est:将选定的行替换为文本 * a est:在行后面追加文本,支持使用 实现多行追加 * i est:在行前面追加文本,支持使用 实现多行追加 * 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选项输出,所以就会输出两次 first first [root@node1 tmp]# echo first | sed -n '1p' #使用-n选项就只输出想要的行,而不是把模式空间中的内容输出 first

[root@node1 tmp]# echo -e "222 222"| sed 's/2/1/' #懒惰
122
122
[root@node1 tmp]# echo -e "222 222"| sed 's/2/1/g' #如果编辑命令后不加g,默认只修改每一行中第一个匹配的值,加上g则表示全局修改(勤奋)
111
111

[root@node1 tmp]# cat a.sh
1
end1
2
end2
3
end3
4
end4
[root@node1 tmp]# sed -n '/1/,/end/p' a.sh
1
end1
[root@node1 tmp]# sed -n '1,/end/p' a.sh #位置匹配是懒惰的,只匹配第一次匹配到的位置
1
end1

[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 walter


<br />
**1.**删除grub的启动项文件中所有以空白开头的行行首的空白字符

sed 's/[1]*//' /etc/grub2.cfg
sed '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' #这个是匹配失败,返回原来的值,匹配不到112
121

[root@node1 tmp]# echo "12 21" | sed -r 's@(ww)s?(w*w)@1=2@' #1是12,2是21
1
2=2*1

$ echo 'this is en example' | sed -E 's@(w+)@[1]@g'
[this] [is] [en] [example]



<br />
**总结:**
`sed`命令按行编辑,拥有pattern space和hold space两种模式空间,日常所用也就pattern space,但是hold space个人觉得就像是汉诺塔一样有第三个柱子,就多出了很多种玩法了。pattern space就相当于只有两个柱子,玩法自然就少了。`vim`文本编辑器里的替换功能和`sed`命令语法一致。所以需要熟练掌握。

<br />
**hold space高级玩法资料:**
[https://www.cnblogs.com/fhefh/archive/2011/11/22/2259097.html](https://www.cnblogs.com/fhefh/archive/2011/11/22/2259097.html)

  1. [:space:] ↩︎

原文地址:https://www.cnblogs.com/dance-walter/p/10338256.html