【linux相识相知】sed命令
在之前的博客中我们介绍了文本三剑客中grep,本次博客就另外一名剑客——sed做出详细的描述,sed真的是一款强大的工具。下面让我们来一起看一下吧!
概述和工作机制
SED的英文全称为Stream EDitor,中文称流编辑器。默认情况下,它会一行一行的读取文件中的内容,在了解其工作原理之前,首先我们得先知道一下几个概念:
1.模式空间(pattern buffer):sed从文件中读取行首先会放到模式空间中进行执行和处理,定义的sed命令都是在这里执行的,默认情况下会逐行的读,逐行的处理,除非你做了行定界。
2.保持空间(hold buffer):在处理完模式空间的一些行的时候,我们有可能需要一个临时的地方去存放模式空间中的内容,这时候就可以将模式空间中的内容放到保持空间了。
初始情况下,模式空间和保持空间都是空的。

1.默认情况下,将从文件中逐行的读取内容至模式空间;
2.默认情况下,模式空间中的内容在处理完成后将会打印到标准输出;
3.sed命令在模式空间中的都是按顺序执行的,除非指定了行定界,否则将在所有的行上面执行;
4.修改后的行被送至标准输出的之后,模式空间将被清空。
基本选项
语法格式:
sed [OPTION]... {script-only-if-no-other-script} [input-file]...
-n:不输出模式空间中的内容至标准输出
#例子1
[root@localhost ~]# sed '' /etc/fstab #
# /etc/fstab
# Created by anaconda on Wed Sep ::
#
# Accessible filesystems, by reference, are maintained under '/dev/disk'
# See man pages fstab(), findfs(), mount() and/or blkid() for more info
#
/dev/mapper/cl-root / xfs defaults
UUID=0ab24855--4d3e-a61d-99ca54711c2c /boot xfs defaults
/dev/mapper/cl-swap swap swap defaults
[root@localhost ~]#
#例子2
[root@localhost ~]# sed -n '' /etc/fstab
[root@localhost ~]#
默认情况下,sed命令会逐行的读取文件中每一行内容至模式空间,再执行执行单引号内的命令,如例子1,单引号内没有任何内容,所以不会对行内容作任何的处理,所以sed会逐行读取文件的内容,然后逐行的显示到标准输出,每打印一行到标准输出,模式空间就会被清空一次。在例子2中,加了-n的选项,会阻止输出到标准输出,所以就不会显示任何的内容。
-e script:指定多个命令
[root@localhost ~]# cat poetry
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
#例子3
[root@localhost ~]# sed -e '1d' -e '10d' poetry
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)Know it will pass,
)And strength you will gain
使用-e选项,可以同时对行作出多个模式匹配,d的意思是删除。这里的例子3,文件poetry是输入,sed读取第一行的内容至模式1空间,然后判断是否是第一行,判断成功是第一行,则删除第一行(从模式空中清除,也不会打印至标准输出),读取第二行,先判断是否是第一行,不是,再判断是否是第十行,不是,然后打印至标准输出,所以在标准输出中只显示了不包含第一行和第十行的其他行的内容。注意:sed不会修改原文件的内容。
-f /path/to/script:把sed的编辑命令放在文件中,注意每一行一个命令:
#例子4
[root@localhost ~]# cat sed_script.txt #编辑命令存放的文本文件
1d
10d
[root@localhost ~]# sed -f sed_script.txt poetry
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)Know it will pass,
)And strength you will gain
我们可以看到例子4中sed使用-f选项指定编辑命令所在的文件,执行的是和例子3同样的操作,删除第一行和第十行,当我们得编辑命令有很多,把它放在文件也是一个不错的做法吧!
行定界
默认情况下不指定行的定界,会读取处理整个文件的每一行。如果指定了行,还是会读取每一行,但是在执行各种命令操作之后,会进行判断是不是定界的行,如果判断成功,则执行操作,如删除行,如果判断失败,则不做任何的处理然后打印至标准输出,下面来看一下几种常见的定界方法:
(1)定界为空,对全文的每一行进行处理
[root@localhost ~]# sed '' poetry
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
[root@localhost ~]# sed '2d' poetry #当判断为第二行的时候,会删除第二行
)Never give up,
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
b./pattern/:被模式匹配到的行,默认可以是基本的正在表达式
[root@localhost ~]# sed '/As/ d' poetry #删除被模式匹配的行
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
c.$最后一行
[root@localhost ~]# sed '$ d' poetry #删除最后一行
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
(3)定界范围
a.x,y:x至y行
[root@localhost ~]# sed '1,9 d' poetry #删除了1-9行
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
b.x,+y:x行到,x行往下数y个行
[root@localhost ~]# sed '1,+3 d' poetry #删除第一行,和第一行往下的3行,也就是一直删除到第四行
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
c.x,/pattren/:x到被模式匹配的行
[root@localhost ~]# sed '2,/smile/ d' poetry #删除第二行至被smile匹配的行,smile被第九行匹配
)Never give up,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
d./pattren1/,/pattern2/:被模式1匹配的行开始到被模式2匹配的行
[root@localhost ~]# sed '/Always/,/put/ d' poetry
)Never give up,
)Never lose hope.
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
[root@localhost ~]# sed '1~2 d' poetry #显示偶数行
)Never lose hope.
)It allows you to cope.
)As they always do.
)Your dreams will come true.
)You'll live through your pain.
)And strength you will gain
[root@localhost ~]# sed '2~2 d' poetry #显示奇数行
)Never give up,
)Always have faith,
)Trying times will pass,
)Just have patience,
)So put on a smile,
)Know it will pass,
编辑命令
下面我们来看一下编辑命令,如前面的d选项是删除行的意思,那么除了d选项,还有哪些呢?
p:显示模式空间的内容
#例5
[root@localhost ~]# sed 'p' poetry
)Never give up,
)Never give up,
)Never lose hope.
)Never lose hope.
)Always have faith,
)Always have faith,
)It allows you to cope.
)It allows you to cope.
)Trying times will pass,
)Trying times will pass,
)As they always do.
)As they always do.
)Just have patience,
)Just have patience,
)Your dreams will come true.
)Your dreams will come true.
)So put on a smile,
)So put on a smile,
)You'll live through your pain.
)You'll live through your pain.
)Know it will pass,
)Know it will pass,
)And strength you will gain
)And strength you will gain
#例6
[root@localhost ~]# sed -n 'p' poetry
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
在例5中,每一行都被打印了2次,原因是在模式空间的时候被打印了一次,然后处理完成之后,在标准输出又被打印了一次,所以自然会显示出两次。如果使用-n,则仅仅打印的模式空间的内容。
a test:在行后面追加文本"test",可以使用\n实现多行追加
[root@localhost ~]# sed '1 a hi i am here\nhey ' poetry
)Never give up,
hi i am here
hey
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
i test:在 行前面插入文本test,支持使用\n实现多行插入
[root@localhost ~]# sed '2i hi i am here\nhey ' poetry
)Never give up,
hi i am here
hey
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
c test:把匹配到的行替换为此处指定的行
[root@localhost ~]# sed '2c hi i am here ' poetry
)Never give up,
hi i am here
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
w /path/to/somefile:保存模式空间中匹配到的内容到指定的文件中
[root@localhost ~]# sed '/So/ w /tmp/poetry' poetry
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
[root@localhost ~]#
[root@localhost ~]# cat /tmp/poetry
)So put on a smile,
r /path/from/somefile:读取指定文件的内容至当前文件中被模式匹配的行后面
[root@localhost ~]# cat poetry2.txt
)yes! I can!
[root@localhost ~]# sed '12 r poetry2.txt' poetry
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
)yes! I can!
=:为模式匹配到的行打印行号,打印在行的上面
[root@localhost ~]# sed '/Never/ =' poetry )Never give up, )Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
!:条件取反
[root@localhost ~]# sed '/Never/ !d' poetry #不删除被默认匹配的行
)Never give up,
)Never lose hope.
s///:查找替换,分割符可以自定义,比如使用s@@@flag或者s###flag,默认会替换被匹配的第一个位置,但是可以添加替换标记:
s///g:全局替换
[root@localhost ~]# cat poetry
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
[root@localhost ~]# sed 's/\<you\>/your/g' poetry
)Never give up,
)Never lose hope.
)Always have faith,
)It allows your to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength your will gain
sed默认支持的基本的正则表达式,使用-r, --regexp-extended选项可以支持扩展的正则表达式。
练习1:删除/boot/grub2/grub.cfg文件中所有以空白字符开头的行的行首的所有空白字符
[root@localhost ~]# sed -r 's@^[[:space:]]+@@g' /boot/grub2/grub.cfg
练习2:删除/etc/fstab文件中所有以#开头的行的行首的#号及#后面的所有空白字符
[root@localhost ~]# sed 's/^#[[:space:]]*//g' /etc/fstab
练习3:输出一个目录给sed,取出其目录,类似于dirname
[root@localhost ~]# echo "/var/log/messages" | sed -r 's@[^/]+/?$@@'
/var/log/
[root@localhost ~]#
[root@localhost ~]# echo "/var/log/messages/" | sed -r 's@[^/]+/?$@@'
/var/log/
高级编辑命令
sed还有一些高级的命令,会用到之前说到的保持空间。

举一些例子来看一下吧!
[root@localhost ~]# cat poetry
)Never give up,
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
可以自己先思考一下答案,再点开:
[root@localhost ~]# sed -n 'n;p' poetry #只显示偶数行
)Never lose hope.
)It allows you to cope.
)As they always do.
)Your dreams will come true.
)You'll live through your pain.
)And strength you will gain
sed -n 'n;p' poetry
[root@localhost ~]# sed '1!G;h;$!d' poetry #逆序显示
)And strength you will gain
)Know it will pass,
)You'll live through your pain.
)So put on a smile,
)Your dreams will come true.
)Just have patience,
)As they always do.
)Trying times will pass,
)It allows you to cope.
)Always have faith,
)Never lose hope.
)Never give up,
sed '1!G;h;$!d' poetry
[root@localhost ~]# sed '$!d' poetry #只保留最后一行,类似tail -
)And strength you will gain
sed '$!d' poetry
[root@localhost ~]# sed '$!N;$!D' poetry #取出最后两行
)Know it will pass,
)And strength you will gain
sed '$!N;$!D' poetry
[root@localhost ~]# sed 'G' poetry #在每一行的后面添加一个空白行
)Never give up, )Never lose hope. )Always have faith, )It allows you to cope. )Trying times will pass, )As they always do. )Just have patience, )Your dreams will come true. )So put on a smile, )You'll live through your pain. )Know it will pass, )And strength you will gain
sed 'G' poetry
这些都是一些高级的用法,在工作中用的可能比较少,因为有的结果完全可以用其他简单的命令去实现。
循环和分支
sed最牛逼之处莫过于这个了,sed提供了一个循环和分支工来控制程序的执行流程。
循环
sed循环的工作原理类似于现代编程语言中的goto语句。
定义sed的标签:
:label
跳转到标签的位置,使用b命令后面跟着标签名即可,下面我们来看一个具体的实例:
[root@localhost ~]# cat teleplay
剧名:白夜追凶
评分:9.0
剧名:秘密深林
评分:9.3
剧名:权利的游戏第七季
评分:9.3
剧名:请回答1988
评分:9.6
[root@localhost ~]# sed -n '
h;n;H;x
s@\n@,@
/请回答/!b label
s@^@---@
:label
p' teleplay
剧名:白夜追凶,评分:9.0
剧名:秘密深林,评分:9.3
剧名:权利的游戏第七季,评分:9.3
---剧名:请回答1988,评分:9.6
分析:
由于命令很长,我们可以将命令分行来写,当然也可以写在一行里面:
1.h;n;H;x:将第一行内容读取到模式空间,执行h,将模式空间的内容覆盖至保持空间,执行n,从文件中读取第二行覆盖至模式空间,执行H,将模式空间的内容追加至保持空间,执行x,将保持空间的内容和模式空间调换;(这里得到的结果就是模式空间中存在两行);
2.s@\n@,@,将换行符替换为逗号;这样原来的两行就变成了一行;
3./请回答/!b label,判断这一行中是否有“请回答”三个字符,如果没有则直接跳到":label",然后执行打印"p",打印模式空间中的内容,如果有则执行"s@^@---@",将在行首添加字符"---"。
这样一来,只有"剧名:请回答1988,评分:9.6"被匹配,然后进行了行首的替换操作。
分支
可以使用t来创建分支。使用t命令跳转到指定的标签,一样我们来看一个例子:
[root@localhost ~]# sed -n '
> h;n;H;x
> s/\n/,/
> :loop
> /白夜追凶/s/^/-/
> /-----/!t loop
> p' teleplay
-----剧名:白夜追凶,评分:9.0
剧名:秘密深林,评分:9.3
剧名:权利的游戏第七季,评分:9.3
剧名:请回答1988,评分:9.6
分析:
由于命令很长,我们可以将命令分行来写,当然也可以写在一行里面:
1:h;n;H;x:将第一行内容读取到模式空间,执行h,将模式空间的内容覆盖至保持空间,执行n,从文件中读取第二行覆盖至模式空间,执行H,将模式空间的内容追加至保持空间,执行x,将保持空间的内容和模式空间调换;(这里得到的结果就是模式空间中存在两行);
2:s@\n@,@,将换行符替换为逗号;这样原来的两行就变成了一行;
3::loop:定义了一个loop标签;
4:/白夜追凶/s/^/-/:是否匹配模式,如果匹配则在其行首添加一个"-";如果不匹配,则直接打印;
5:/-----/!t loop:是否存在5个"-",如果不存在,则跳到标签loop处,继续执行第4步添加"-",直到满足5个"-",则跳出循环打印。
最后,我们之前的所有的操作,都是没有修改文件的本身的,可以使用 -i 选项来直接修改文件本身,慎用,建议在使用之前,先备份一下文件!
[root@localhost ~]# sed -i '1d' poetry
[root@localhost ~]#
[root@localhost ~]# cat poetry
)Never lose hope.
)Always have faith,
)It allows you to cope.
)Trying times will pass,
)As they always do.
)Just have patience,
)Your dreams will come true.
)So put on a smile,
)You'll live through your pain.
)Know it will pass,
)And strength you will gain
总结:sed的确是一个强大的文本处理工具,功能非常丰富,需要在今后的日常使用和工作中不断的熟悉和巩固。
参考链接:http://blog.jobbole.com/109088/ 《三分钟学会SED》写的非常好!
【linux相识相知】sed命令的更多相关文章
- Linux实战教学笔记12:linux三剑客之sed命令精讲
第十二节 linux三剑客之sed命令精讲 标签(空格分隔): Linux实战教学笔记-陈思齐 ---更多资料点我查看 1,前言 我们都知道,在Linux中一切皆文件,比如配置文件,日志文件,启动文件 ...
- Linux中使用sed命令或awk命令修改常规配置文件
一.方案: Linux中使用sed命令或awk命令修改常规配置文件 二.步骤: 1.假设有一个a.txt,内容如下: #!/bin/bash aa= bbb= ccc= #ddd= 2.如果想要把里面 ...
- day14 linux三剑客之sed命令
day14 linux三剑客之sed命令 sed命令 Sed 主要用来自动编辑一个或多个文件.简化对文件的反复操作.编写转换程序等. sed(流式编辑器) : sed主要用来修改文件. 1.sed命令 ...
- 【Linux相识相知】计算机的组成、linux发行版和哲学思想、基础命令和目录结构(FHS)
从今天开始,Frank将开始在博客上记录自己学习linux的点点滴滴,F初来乍到,还望各位大佬多多指教.本次博客的主要内容如下: 计算机基础:简要的描述了计算机的组成及其功能: linux初识:介绍了 ...
- linux shell 用sed命令在文本的行尾或行首添加字符
转自 http://www.cnblogs.com/aaronwxb/archive/2011/08/19/2145364.html 昨天写一个脚本花了一天的2/3的时间,而且大部分时间都耗在了sed ...
- linux三剑客之sed命令
一.前言 我们都知道,在Linux中一切皆文件,比如配置文件,日志文件,启动文件等等.如果我们相对这些文件进行一些编辑查询等操作时,我们可能会想到一些vi,vim,cat,more等命令.但是这些命令 ...
- 快速理解linux流编辑器sed命令
原创 杜亦舒性能与架构 之前介绍过 awk 命令,sed 命令同样是非常重要的文本处理工具,涉及到linux shell开发时,几乎是避不开这两大利器的 sed 是 stream editor 的简写 ...
- Linux Shell编程 sed命令
概述 sed 是一种几乎可以应用在所有 UNIX 平台(包括 Linux)上的轻量级流编辑器,体积小.所以,它可以对从如管道这样的标准输入中接收的数据进行编辑. sed 主要是用来将数据进行选取.替换 ...
- Linux学习之sed命令详解
概述 sed是stream editor的简称,也就是流编辑器.它一次处理一行内容,处理时,把当前处理的行存储在临时缓冲区中,称为“模式空间”(pattern space),接着用sed命令处理缓冲区 ...
随机推荐
- opencv3.3.1+vs2015+c++实现直接在图像上画掩码,保存掩码图片
左键红右键蓝,保存为k #include "opencv2\imgproc\imgproc.hpp" // Gaussian Blur #include "opencv2 ...
- 201621123012 《java程序设计》第3周学习总结
1. 本章学习总结 1.1 写出你认为本周学习中比较重要的知识点关键词 答:关键词:继承 多态 抽象类 abstract 覆盖 object siper 1.2 尝试使用思维导图将这些关键词组织起来. ...
- audio音频
html5的audio功能上已经非常强大,回放,跳转,缓冲等以前只能用flash才能实现的功能,html5的audio都能轻松搞定 最近的一个项目使用到了这个功能,把我使用的情况写下来,供大家参考, ...
- Ubuntu 安装后的配置及美化(一)
Ubuntu 安装后的配置及美化(一) 记录一下 完成后的主界面. 配置 1.更新源为阿里云 找到 软件和更新 选项,更新源为阿里云的源. 在 其他软件 中将 Canonical合作伙伴 打上勾. 然 ...
- Opencv博文收藏列表
opencv识别二维码:https://blog.csdn.net/jia20003/article/details/77348170 opencv视频:http://www.opencv.org.c ...
- (转)取消目录与SVN的关联
第一种方法: 直接.逐级地删除目标目录中的隐藏属性的.svn目录 第二种方法: 如果用的是TortoiseSVN客户端,则先在另外一处建立一个新目录A,右键点住svn目录并拖动到A上松手,在弹出的菜单 ...
- YY的GCD 数学
题目描述 神犇YY虐完数论后给傻×kAc出了一题 给定N, M,求1<=x<=N, 1<=y<=M且gcd(x, y)为质数的(x, y)有多少对 kAc这种傻×必然不会了,于 ...
- Linux下指定线程的名字
为了能方便的区分一个进程中的每个线程,可以通过prctl()给每个线程取个名字.这样在会创建多个线程的程序执行过程中,就能知道一个pid或tid对应的是哪个线程,对调试程序有一定帮助. prctl是个 ...
- 19年PDYZ冬令营游记
我和卓越的那些事 ——2019年平度一中卓越计划冬令营 题前记: 正月十三那天,刚看完<流浪地球>,便接到了一个电话,老妈告诉我竟然一中组织了一个冬令营,并且起了一个很好的名字“卓越计 ...
- [o.e.b.ElasticsearchUncaughtExceptionHandler] [unknown] uncaught exception in thread [main] org.elasticsearch.bootstrap.StartupException: java.lang.RuntimeException: can not run elasticsearch as root
这是因为不知从哪个版本后,elasticsearch为了安全性,是不能用root用户启动的. 解决的办法:当然是创建一个用户,用创建的用户启动啦,注意权限的问题,目录也应该改为创建的用户权限! 我是用 ...