[TOC]
sed命令详解     sed:Stream Editor文本流编辑,sed是一个“非交互式的”面向字符流的编辑器。能同时处理多个文件多行的内容,可以不对原文件改动,把整个文件输入到屏幕,可以把只匹配到模式的内容输入到屏幕上。还可以对原文件改动,但是不会再屏幕上返回结果。
sed命令的语法格式:  
sed命令的选项(option): 1 2 3 4 5 -n :只打印模式匹配的行 -e :直接在命令行模式上进行sed动作编辑,此为默认选项 -f :将sed的动作写在一个文件内,用–f filename 执行filename内的sed动作 -r :支持扩展表达式 -i :直接修改文件内容 
 
-f :将sed的动作写在一个文件内,用–f filename 执行filename内的sed动作
-r :支持扩展表达式
-i :直接修改文件内容
sed在文件中查询文本的方式: 1)使用行号,可以是一个简单数字,或是一个行号范围 
x 
x为行号 
 
 
x,y 
表示行号从x到y 
 
/pattern 
查询包含模式的行 
 
/pattern /pattern 
查询包含两个模式的行 
 
pattern/,x 
在给定行号上查询包含模式的行 
 
x,/pattern/ 
通过行号和模式查询匹配的行 
 
x,y! 
查询不包含指定行号x和y的行 
 
 
 
使用正则表达式、扩展正则表达式(必须结合-r选项) 
^ 
锚点行首的符合条件的内容,用法格式”^pattern” 
 
 
$ 
锚点行首的符合条件的内容,用法格式”pattern$” 
 
^$ 
空白行 
 
. 
匹配任意单个字符 
 
* 
匹配紧挨在前面的字符任意次(0,1,多次) 
 
.* 
匹配任意长度的任意字符 
 
\? 
匹配紧挨在前面的字符0次或1次 
 
\{m,n\} 
匹配其前面的字符至少m次,至多n次 
 
\{m,\} 
匹配其前面的字符至少m次 
 
\{m\} 
精确匹配前面的m次\{0,n\}:0到n次 
 
\< 
锚点词首——相当于 \b,用法格式:\<pattern 
 
> 
锚点词尾,用法格式:>pattern 
 
\ 
单词锚点 
 
 
分组,用法格式:pattern,引用\1,\2 
 
[] 
匹配指定范围内的任意单个字符 
 
 
匹配指定范围外的任意单个字符 
 
[:digit:] 
所有数字, 相当于0-9, [0-9]—-> [[:digit:]] 
 
[:lower:] 
所有的小写字母 
 
[:upper:] 
所有的大写字母 
 
[:alpha:] 
所有的字母 
 
[:alnum:] 
相当于0-9a-zA-Z 
 
[:space:] 
空白字符 
 
[:punct:] 
所有标点符号 
 
 
 
sed的匹配模式支持正则表达式 1 2 3 4 5 6 sed'5 q' /etc/ passwd sed-n '/r*t/p' /etc/ passwd sed-n '/.r.*/p' /etc/ passwd sed-n '/o*/p' /etc/ passwd sed-n '/o\{1,\}/p' /etc/ passwd sed-n '/o\{1,3\}/p' /etc/ passwd 
 
转义bai符\的作用是消除有特殊含义字符du的特殊意义,使其还原为普通字符。 
sed的编辑命令(sed command): 
p 
打印匹配行(和-n选项一起合用) 
 
 
= 
显示文件行号 
 
a\ 
在定位行号后附加新文本信息 
 
i\ 
在定位行号后插入新文本信息 
 
d 
删除定位行 
 
c\ 
用新文本替换定位文本 
 
w filename 
写文本到一个文件,类似输出重定向 > 
 
r filename 
从另一个文件中读文本,类似输入重定向 < 
 
s 
使用替换模式替换相应模式 
 
q 
第一个模式匹配完成后退出或立即退出 
 
l 
显示与八进制ACSII代码等价的控制符 
 
{} 
在定位行执行的命令组,用分号隔开 
 
n 
从另一个文件中读文本下一行,并从下一条命令而不是第一条命令开始对其的处理 
 
N 
在数据流中添加下一行以创建用于处理的多行组 
 
g 
将模式2粘贴到/pattern n/ 
 
y 
传送字符,替换单个字符 
 
 
 
对文件的操作无非就是”增删改查“,怎样用sed命令实现对文件的”增删改查“,玩转sed是写自动化脚本必须的基础之一。 
1 2 3 4 5 6 7 sed -n '/^#/!p'   /etc/ vsftpd/vsftpd.conf          sed -n '/^#/!{/^$/!p}'   /etc/ vsftpd/vsftpd.conf   sed -e '/^#/d'  -e '/^$/d'   /etc/ vsftpd/vsftpd.conf   sed -n '1,/adm/p'  /etc/ passwd   sed -n '/adm/,6p'  /etc/ passwd   sed -n '/adm/,4p'  /etc/ passwd   sed -n '/adm/,2p'  /etc/ passwd   
 
sed命令实现对文件内容的添加:(对源文件添加的话就用-i参数): 
1 2 3 4 5 6 7 8 9 #1)匹配行的行首添加,添加在同行   #2)匹配行的行中的某个字符后添加   #3)匹配行的行尾添加字符   #4)匹配行的行前面行添加   #5)匹配行的行后面行添加   #6)文件的行首添加一行     [root@jie1 ~]# sed -i '1 i\sed command start'  myfile   #7)文件的行尾追加一行   [root@jie1 ~]# sed -i '$a \sed command end'  myfile   
 
重点:sed命令实现对文件内容的替换(替换是在shell自动化脚本中用到最多的操作) 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 [root@jie1  ~] anonymous_enable=YES write_enable=YES local_umask=022 xferlog_enable=YES connect_from_port_20=YES root: x: 0: 0:root :/root :/bin/bash bin: x: 1:1 :bin :/bin :/sbin/nologin daemon: x: 2:2 :daemon :/sbin :/sbin/nologin adm: x: 3:4 :adm :/var/adm :/sbin/nologin lp: x: 4:7 :lp :/var/spool/lpd :/sbin/nologin DEVICE="eth0"  BOOTPROTO="static"  HWADDR="00:0C:29:90:79:78"  ONBOOT="yes"  IPADDR=172.16 .22.1  NETMASK=255.255 .0.0 [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~]          [root@jie1  ~] anonymous_enable=YES write_enable=YES local_umask=022 xferlog_enable=YES connect_from_port_20=NO (root):x : 1:1 : (root):/ (root):/bin/nologin  bin: x: 1:1 :tom :/tom :/stom/nologin daemon: x: 2:2 :jerry :/sbin :/stom/nologin daemon: x: 2:2 :jerry :/sbin :/stom/nologin adm: x: 3:4 :boss :/var/adm :/sbin/nologin lp: x: 4:7 :lp :/var/spool/lpd :/sbin/nologin Ethernet HWADDR="00:0C:29:90:79:78"  ONBOOT="yes"  IPADDR=172.16 .10.12  NETMASK=255.255 .0.0 
 
sed引用变量:(在自动化shell脚本 中也经常会使用到变量) 1 2 3 4 5 6 7 8 9   sed -i '2s/node_base/' "$i " '/'  /etc/libvirt/qemu/$i .xml sed -i "2s/node_base/$i /"   /etc/libvirt/qemu/$i .xml sed   "/device_name/s/$newname /$oldname /"  $file /new_dev.cfg  //在含有device_name的行中将$newname 替换为$oldname  
 
 第二种当sed命令里面有默认的变量时,那自己定义的变量需要加单引号,且sed里面的语句必须用单引 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 [root@jie1  ~]# cat >> myfile << EOF  > hello world > i am jie > how are you > EOF   #先生成一个文件  [root@jie1  ~]# cat myfile  hello world i am jie how are you [root@jie1  ~]# name=li           #定义一个变量,且给变量赋值  [root@jie1  ~]# sed -i "s/jie/$name/"  myfile           #把匹配jie的字符替换成变量的值  [root@jie1  ~]# cat myfile  hello world i am li how are you [root@jie1  ~]# sed -i "$a $name"  myfile            #当sed命令也有默认变量时,在去引用自己定义的变量会出现语法错误  sed: -e expression #1, char 3: extra characters after command  [root@jie1  ~]# sed -i '$a ' $name''  myfile            #在引用自定义的变量时,sed语句必须用单引引住,然后把自定义的变量也用单引号引住  [root@jie1  ~]# cat myfile  hello world i am li how are you li [root@jie1  ~]#  
 
读取一个文件到正在用sed操作的文件中 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 [root@jie1 ~]# cat myfile   #文件内容 hello world i am li how are you li [root@jie1 ~]# cat test  #将用sed操作的文件的内容 Ethernet HWADDR ="00:0C:29:90:79:78" ONBOOT ="yes" IPADDR =172.16.10.12NETMASK =255.255.0.0[root@jie1 ~]# sed  -i '/Ethernet/r myfile'  test       #在匹配Ethernet的行,读进来另一个文件的内容,读进来的文件的内容会插入到匹配Ethernet的行后 [root@jie1 ~]# cat test  #再次查看用sed命令操作的行 Ethernet hello world i am li how are you li HWADDR ="00:0C:29:90:79:78" ONBOOT ="yes" IPADDR =172.16.10.12NETMASK =255.255.0.0[root@jie1 ~]# 
 
sed的经典例子: 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 ##1 )、处理以下文件内容,将域名取出并进行计数排序,如处理: http://www.baidu.com /index .html http://www.baidu.com /1 .html http://post.baidu.com /index .html http://mp3.baidu.com /index .html http://www.baidu.com /3 .html http://post.baidu.com /2 .html 得到如下结果: 域名的出现的次数 域名 3  www.baidu.com 2  post.baidu.com 1  mp3.baidu.com [root@localhost shell ]# cat  file  | sed -e  ' s/http:\/\///'  -e  ' s/\/.*//'  | sort  | uniq -c  | sort  -rn 3  www.baidu.com 2  post.baidu.com 1  mp3.baidu.com [root@codfei4 shell ]# awk -F/ '{print $3}'  file  |sort  -r|uniq -c |awk '{print $1"\t",$2}'  3  www.baidu.com 2  post.baidu.com 1  mp3.baidu.com ##2 )、用grep 结合sed取出网卡的ip地址 [root@jie1 ~]# ifconfig | grep  -B1 "inet addr"  |grep  -v "\-\-"  |sed -n -e  'N;s/\(eth[0-9]\).*\n.*addr:\([0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}\).*/\1 \2/p'  
 
参考: https://blog.csdn.net/weixin_40572607/article/details/90812959