linux下sed的使用+練習

koudaidai發表於2012-08-27

sed是款非互動式的編輯器,它沒有破壞性,不會修改檔案,除非使用重定向來儲存輸出結果,預設
情況下,所有的輸出行都被列印在螢幕上。
sed把當前正在處理的行儲存在一個臨時緩衝區中,這個緩衝區被稱為模式空間或臨時緩衝。sed處理
完模式空間中的行後,就把該行傳送到螢幕上。每處理完一行就將其從模式空間中刪除,然後將下一
行讀入空間,進行處理和顯示。 sed把每一行都存在臨時緩衝區中,對這個副本進行編輯,所以不會
修改或破壞原檔案。
下面是sed的命令和選項。
a\                        在當前行後新增一行或多行
c\                         用新文字修改(替換)當前行中的文字
d                          刪除行
i\                          在當前行之前插入文字
h                          把模式空間裡的內容複製到暫存緩衝區
H                          把模式空間裡的內容追加到暫存緩衝區
g                           取出暫存緩衝區的內容,將其複製到模式空間,覆蓋該處原有內容
G                          取出暫存緩衝區的內容,將其複製到模式空間,追加到原有內容後面
l                            列出非列印字元
p                           列印行
n                           讀入下一輸入行,並從下一條命令而不是第一條命令開始對其的處理
q                           結束或退出sed
r                            從檔案中讀取輸入行
!                            從所選行以外的所有行應用命令
s                            用一個字串替換另一個
替換標誌
g                            在行內進行全域性替換
p                            列印行
w                            將行寫入檔案
x                             交換暫存緩衝區與模式空間的內容
y                             將字元轉換為另一字元

選項
-e                            允許多項編輯
-f                             指定sed指令碼檔名
-n                            取消預設的輸出
如果需要使用多條命令,或需要在某個地址範圍內巢狀地址,就必須用花括號將命令括起來,每行
只寫一條命令,或用分號分隔同一行中的多條命令。

緊跟在s命令後的字元就是查詢串和替換串之間的分隔符。分隔符預設為正斜槓,但可以改變。無論
什麼字元(換行符,反斜線除外),只要緊跟s命令,就成了新的串分隔符。這個方法在查詢包含正
斜槓的模式時很管用,例如查詢路徑名或生日。

sed支援的元字元
^                               行首定位符                 /^love/                   匹配所有love開頭的行
$                              行尾定位符                 /love$/                  匹配所有以love結尾的行
.                               匹配除換行符
                                之外的單個字元          /l..e/                      匹配包含l,後跟兩個任意字元,再
                                                                                                跟字母e的行
*                              匹配0或多個前字元     /*love/                 匹配在0或多空格緊跟著模式love的行
[]                             匹配指定字元組內的                            
                               任一字元                     [lL]ove/                匹配包含love和Love的行
[^]                            匹配不再指定字元組
                               內的任一字元             /[^A-KM-Z]ove/      匹配包含ove但之前的字元不在A至K或
                                                                                              M至Z之間的行
\(..\)                         儲存已匹配的字元     s/\(love\)able/\1er/      loveable被替換為lover
&                            儲存查詢串以便在
                               替換串中引用             s/love/**&**/             love變成了**love**
\<                            詞首定位符                 /\<love/                   匹配包含以love開頭的單詞的行
\>                            詞尾定位符                 /love\>/                   匹配包含以love結尾的單詞的行
x\{m\}                      連續m個x                    /o\{5\}/
x\{m,\}                     至少m個x                    /o\{5,\}/
x\{m,n\}                   至少m個,但不超過n個    /o\{5,10\}/


練習回顧
sed -n '/sentimental/p' filex                    把檔案filex中所有包含sentimental的行列印在螢幕上
                                                              filex的內容不會被改變 如果沒有-n選項,所有包含
                                                              sentimental的行都會被列印兩次
sed '1,3d' filex > newfilex                       刪除檔案filex的前3行,將修改的結果儲存在newfilex檔案中
sed '/[Dd]aniel/d' filex                            刪除包含Daniel或daniel的行
sed -n '15,20p' filex                               只列印第15~20行
sed '1,10s/Montana/MT/g' filex              將1~10行所有Montana全域性替換為MT(如果沒有g替換選項
                                                                 將只替換每行中的第一個Montana)
sed '/March/!d' filex(sh)
sed '/March/\!d' filex(csh)                       刪除所有不含March的行(只在csh中才要用反斜槓轉義)
sed '/report/s/5/8' filex                            把所有包含report的行裡出現的第一個5改為8
sed 's/...$//' filex                                     刪除每行的後3個字元
sed 's/...//' filex                                       刪除每行的前3個字元
sed '/east/,/west/s/North/South/' filex     把從east到west這個範圍內所有行出現的North替換為South
sed -n '/Time off/w timefile' filex              將所有包含Time off的行寫入到timefile檔案中
sed 's/\([Oo]ccur\)ence/\1rence/' file      將所有Occurence替換成Occurence,occurence替換成occurence
sed -n '|' filex                                           列印所有以\nn顯示非列印字元,製表鍵(tab)顯示為>的行

[root@linux ~]# sed [-nefr] [動作]

練習

範例一∶將 /etc/passwd 的內容列出,並且我需要列印行號,同時,請將第 2~5 行刪除!
[root@linux ~]# nl /etc/passwd | sed '2,5d'
1 root:x:0:0:root:/root:/bin/bash
6 sync:x:5:0:sync:/sbin:/bin/sync
7 shutdown:x:6:0:shutdown:/sbin:/sbin/shutdown
.....(後面省略).....
# 看到了吧?因為 2-5 行給他刪除了,所以顯示的資料中,就沒有 2-5 行棉~
# 另外,注意一下,原本應該是要下達 sed -e 才對,沒有 -e 也行啦!
# 同時也要注意的是, sed 後面接的動作,請務必以 '' 兩個單引號括住喔!
# 而,如果只要刪除第 2 行,可以使用 nl /etc/passwd | sed '2d' 來達成,
# 至於第 3 到最後一行,則是 nl /etc/passwd | sed '3,$d' 的啦!

範例二∶承上題,在第二行後(亦即是加在第三行)加上『drink tea?』字樣!
[root@linux ~]# nl /etc/passwd | sed '2a drink tea'
1 root:x:0:0:root:/root:/bin/bash
2 bin:x:1:1:bin:/bin:/sbin/nologin
drink tea
3 daemon:x:2:2:daemon:/sbin:/sbin/nologin
# 嘿嘿!在 a 後面加上的字串就已將出現在第二行後面棉!那如果是要在第二行前呢?
# nl /etc/passwd | sed '2i drink tea' 就對啦!

範例三∶在第二行後面加入兩行字,例如『Drink tea or .....』『drink beer?』
[root@linux ~]# nl /etc/passwd | sed '2a Drink tea or ....../
> drink beer ?'
1 root:x:0:0:root:/root:/bin/bash
2 bin:x:1:1:bin:/bin:/sbin/nologin
Drink tea or ......
drink beer ?
3 daemon:x:2:2:daemon:/sbin:/sbin/nologin
# 這個範例的重點是,我們可以新增不只一行喔!可以新增好幾行~
# 但是每一行之間都必須要以反斜線 / 來進行新行的增加喔!所以,上面的例子中,
# 我們可以發現在第一行的最後面就有 / 存在啦!那是一定要的喔!

範例四∶我想將第2-5行的內容取代成為『No 2-5 number』呢?
[root@linux ~]# nl /etc/passwd | sed '2,5c No 2-5 number'
1 root:x:0:0:root:/root:/bin/bash
No 2-5 number
6 sync:x:5:0:sync:/sbin:/bin/sync
# 沒有了 2-5 行,嘿嘿嘿嘿!我們要的資料就出現啦!

範例五∶僅列出第 5-7 行
[root@linux ~]# nl /etc/passwd | sed -n '5,7p'
5 lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin
6 sync:x:5:0:sync:/sbin:/bin/sync
7 shutdown:x:6:0:shutdown:/sbin:/sbin/shutdown
# 為什麼要加 -n 的引數呢?您可以自行下達 sed '5,7p' 就知道了!(5-7行會重複輸出)
# 有沒有加上 -n 的引數時,輸出的資料可是差很多的喔!

範例六∶我們可以使用 ifconfig 來列出 IP ,若僅要 eth0 的 IP 時?
[root@linux ~]# ifconfig eth0
eth0 Link encap:Ethernet HWaddr 00:51:FD:52:9A:CA
inet addr:192.168.1.12 Bcast:192.168.1.255 Mask:255.255.255.0
inet6 addr: fe80::250:fcff:fe22:9acb/64 Scope:Link
UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1
.....(以下省略).....
# 其實,我們要的只是那個 inet addr:..那一行而已,所以棉,利用 grep 與 sed 來捉
[root@linux ~]# ifconfig eth0 | grep 'inet '|sed 's/^.*addr://g'|sed 's/Bcast.*$//g'
[root@linux ~]# ifconfig eth0 | grep 'inet '|sed 's/^.*addr://g'|sed 's/ Bcast.*$//g'
#這兩行命令差不多,但是第二行在Bacst前有個空格,一定要注意,說明連空格在內都去除了,可以使用
#上面兩行命令重向到兩個文字檔案中,例如1.txt 2.txt 仔細檢視這兩個檔案的大小
# 您可以將每個管線 (|) 的過程都分開來執行,就會曉得原因棉!
# 去頭去尾之後,就會得到我們所需要的 IP 亦即是 192.168.1.12 棉~

範例七∶將 /etc/man.config 檔案的內容中,有 MAN 的設定就取出來,但不要說明內容。
[root@linux ~]# cat /etc/man.config | grep 'MAN'| sed 's/#.*$//g'| sed '/^$/d'
# 每一行當中,若有 # 表示該行為註解,但是要注意的是,有時候,
# 註解並不是寫在第一個字元,亦即是寫在某個指令後方,如底下的模樣∶
# 『shutdown -h now # 這個是關機的指令』,註解 # 就在指令的後方了。
# 因此,我們才會使用到將 #.*$ 這個正規表示法!

範例八∶利用 sed 直接在 ~/.bashrc 最後一行加入『# This is a test』
[root@linux ~]# sed -i '$a # This is a test' ~/.bashrc
# 上頭的 -i 引數可以讓你的 sed 直接去修改後面接的檔案內容喔!而不是由螢幕輸出。
# 至於那個 $a 則代表最後一行才新增的意思。

相關文章