sed 高级用法(1) 之 n N

前言

我在参考其它多个Blog中发现,有些Blog中sed的功能介绍和实际有出入,这可能和版本有关系,正如sed文档中所说“might change in future versions”,所以本文sed是以4.2.1为例。如果以后sed更新版本了,有可能会不一样。

好记性不如烂笔头!

sed的工作原理

sed维护着两个数据的缓冲空间,一个是模式空间(pattern space)和另外一个保留空间(hold space),在初始环境下都为空。

sed是一个流编辑器,它会循环的从输入流中读取每一行,直到读完整个文件。具体如下:

首先,它会从输入流中读取一行(如果刚开始就是第一行),移除行尾的换行符,放置于模式空间当中,接着一条条的运行命令(命令可以有多个并且是按序执行,如果某个命令地址定界了一个行号,只有满足该行号才会执行命令,如“1d”,意思是如果是第一行则删除模式空间内的内容)。

当命令运行完毕之后,除非使用了 -n 选项,否则会把模式空间的内容加上之前删过的换行符并打印到输出。然后读入下行,执行下一个循环。如果没有使诸如‘D’的特殊命令,那会在两个循环之间清空模式空间,但不会清空保留空间。

意译自http://www.gnu.org/software/sed/manual/sed.html中 3.1 How sed Works

流程图如下

sed 工作原理.jpg

注:上面只是标准流程,某些特殊命令会有自己的流程


n N的说明

范例文件1

[root@CZ tmp]# cat 1
1
2
3
4
5
6
7
8
9
10
11

 

n:打印当前模式空间内容,然后读取下一行并替代当前模式空间的内容。如果读取不到下一行sed则会不运行之后的命令

我们通过以下命令了解一下n

[root@CZ tmp]# sed 'n;d' 1
1
3
5
7
9
11

上面命令过程是这样

  1. 先读取第一行进模式空间(以后简称为1)

  2. 执行命令n,过程如下

    1. 打印1到输出

    2. 读取2并覆盖到模式空间

  3. 执行命令d,过程如下

    1. 删除模式空间的内容

    2. 立即执行下一循环(d命令在运行后会直接执行下一循环,所以它并不会执行之后的命令和打印模式空间,具体d介绍会留在下次分享)

  4. 按照上面的流程循环执行…….直到读取到11(最后一行),11的具体过程如下

    1. 读取11进模式空间

    2. 运行命令n,不过读取不到下一行

    3. 因为读不到,所以sed退出所有的命令,也就是说它不会执行命令d

    4. 加回换行符并打印模式空间的内容到输出,当前模式空间内容为11,所以输出11

  5. 已经是文件尾,sed结束运行。 

 

N:读取下一行并且附加到当前模式空间内,如果读取不到下一行sed则会不运行之后的命令

我们通过以下命令了解一下N

[root@CZ tmp]# sed 'N;a---' 1
1
2
---
3
4
---
5
6
---
7
8
---
9
10
---
11

上面命令过程是这样

  1. 读取1进模式空间

  2. 执行命令N

    1. 读取2并附加到模式空间,当前模式空间内容为“1\n2”

  3. 执行命令a—

    1. 在模式空间后附加一行‘—’当前模式空间内容为“1\n2\n—"

  4. 打印模式空间内容

  5. 循环执行直到读取11进模式空间,11的具体过程如下

    1. 读取11进模式空间

    2. 执行命令N,不过读取不到下一行

    3. 因为读不到,所以sed退出所有的命令,也就是说它不会执行命令a

    4. 加回换行符并打印模式空间的内容到输出,当前模式空间内容为11,所以输出11

  6. 已经是文件尾,sed结束运行。 

尾言

n N经常和d D一起用,不过D有点复杂,所以留在下次分享,不过我先剧透一下,D会删除模式空间内第一行,并且如果模式空间内容不为空,它会循环执行前面命令。直到为空才会执行下一循环。


参考

http://www.gnu.org/software/sed/manual/sed.html

上面是官方文档,如果下面和上面冲突,请以上面为准,下面blog有些内容和我理解有出入,请自行判断。

http://www.cnblogs.com/fhefh/archive/2011/11/14/2248942.html

http://www.cnblogs.com/theCambrian/p/3606214.html    
http://blog.csdn.net/yanquan345/article/details/19613443

http://www.cnblogs.com/fhefh/archive/2011/11/22/2259097.html


 

原创文章,作者:Unknown,如若转载,请注明出处:http://www.178linux.com/5900

(0)
UnknownUnknown
上一篇 2015-07-03
下一篇 2015-07-03

相关推荐

  • N21天天第十一周课程练习

    1、详细描述一次加密通讯的过程,结合图示最佳。 SSL协议基础: SSL协议位于TCP/IP协议与各种应用层协议之间,本身又分为两层: 1)SSL记录协议:建立在可靠传输层协议(TCP)之上,为上层协议提供数据封装、压缩、加密等基本功能。 2)SSL握手协议:在SSL记录协议之上,用于实际数据传输前,通讯双方进行身份认证、协商加密算法、交换加密密钥等。 SS…

    Linux干货 2016-10-09
  • python面向对象第二周魔术方法详解

    魔法方法及其使用__开头和结束的方法,定义外部没有办法直接调用,但会有影响使用运算符号的魔法方法, + ,-,*,/,%,//,**, __add__,__sub__,__mul__,__truediv__,__mod__,__floordiv__,__pow__,__divmod__(?),系统内部对于数值型,字符串型,容器内型都定义了其中部分或者全部的运…

    Linux干货 2017-11-22
  • N25 – Week 4 blog

    第四周博客作业了,感觉前面的有好多都忘掉了,但是没有时间补以前的啊……好郁闷,幸好今天周五,明天又是圣诞节,从马云爸爸买了一颗圣诞树,今晚回家装上 1. 复制/etc/skel目录为/home/tuser1,要求/home/tuser1及其内部文件的属组和其他用户均没有任何访问权限 [root@dhcp-10-129-6-166&nb…

    Linux干货 2016-12-20
  • 马哥教育网络班22期第一周课程练习

    1.描述计算机的组成及其功能     计算机的组成部分分为硬件部分与软件部分         硬件部分: I/O设备 + 运算器 + 存储器 + 控制器       &n…

    Linux干货 2016-08-15
  • Linux中的网络功能概念介绍(一)

      在操作系统中,进程之间的通信是用ipc(内存基本通信)以及内存共享shm来实现本地的通信,那么如果跨主机间通信或者说是不同之间的主机用rpc(远程过程调用)以及socket套接字,用一个主机在socket中写数据,另一个主机在socket之间读数据,以实现进程通信的最终目的。   在网络功能出现之前,基本都是用电话…

    Linux干货 2016-11-23
  • 用户和用户组相关的配置文件

    用户和用户组相关的配置文件 一、与用户相关的配置文件 一般来说,与用户配置相关的几个文件如下: l  /etc/passwd: 最重要的文件,存储着用户的用户名,UID,Shell等信息 l  /etc/shadow: 用户密码文件,使用sha-1算法加密存储(注意该文件的权限) l  /etc/skel/: 用户的模板文件,新…

    Linux干货 2016-10-23