溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Linux中如何使用sed文本處理命令

發(fā)布時(shí)間:2021-06-25 15:35:33 來源:億速云 閱讀:174 作者:Leah 欄目:系統(tǒng)運(yùn)維

Linux中如何使用sed文本處理命令,相信很多沒有經(jīng)驗(yàn)的人對(duì)此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個(gè)問題。

sed對(duì)文本的處理很強(qiáng)大,并且sed非常小,參數(shù)少,容易掌握,他的操作方式根awk有點(diǎn)像。sed按順序逐行讀取文件。然后,它執(zhí)行為該行指定的所有操作,并在完成請(qǐng)求的修改之后的內(nèi)容顯示出來,也可以存放到文件中。完成了一行上的所有操作之后,它讀取文件的下一行,然后重復(fù)該過程直到它完成該文件。在這里要注意一點(diǎn),源文件(默認(rèn)地)保持不被修改。sed 默認(rèn)讀取整個(gè)文件并對(duì)其中的每一行進(jìn)行修改。說白了就是一行一行的操作。

參數(shù)

sed -h
 -n, --quiet, --silent    取消自動(dòng)打印模式空間
 -e 腳本, --expression=腳本   添加“腳本”到程序的運(yùn)行列表
 -f 腳本文件, --file=腳本文件  添加“腳本文件”到程序的運(yùn)行列表
 --follow-symlinks    直接修改文件時(shí)跟隨軟鏈接
 -i[擴(kuò)展名], --in-place[=擴(kuò)展名]    直接修改文件(如果指定擴(kuò)展名就備份文件)
 -l N, --line-length=N   指定“l(fā)”命令的換行期望長(zhǎng)度
 --posix  關(guān)閉所有 GNU 擴(kuò)展
 -r, --regexp-extended  在腳本中使用擴(kuò)展正則表達(dá)式
 -s, --separate  將輸入文件視為各個(gè)獨(dú)立的文件而不是一個(gè)長(zhǎng)的連續(xù)輸入
 -u, --unbuffered  從輸入文件讀取最少的數(shù)據(jù),更頻繁的刷新輸出
 --help     打印幫助并退出
 --version  輸出版本信息并退出

例1
測(cè)試文件

代碼如下:


root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/bin/false
daemon:x:2:2:daemon:/sbin:/bin/false
mail:x:8:12:mail:/var/spool/mail:/bin/false
ftp:x:14:11:ftp:/home/ftp:/bin/false
&nobody:$:99:99:nobody:/:/bin/false
zhangy:x:1000:100:,,,:/home/zhangy:/bin/bash
http:x:33:33::/srv/http:/bin/false
dbus:x:81:81:System message bus:/:/bin/false
hal:x:82:82:HAL daemon:/:/bin/false
mysql:x:89:89::/var/lib/mysql:/bin/false
aaa:x:1001:1001::/home/aaa:/bin/bash
ba:x:1002:1002::/home/zhangy:/bin/bash
test:x:1003:1003::/home/test:/bin/bash
@zhangying:*:1004:1004::/home/test:/bin/bash
policykit:x:102:1005:Po


例a,這個(gè)例子,把test文件中的root替換成tankzhang,只不過只替換一次及終止在這一行的操作,轉(zhuǎn)到下一行

代碼如下:


[zhangy@BlackGhost mytest]# sed 's/root/tankzhang/' test |grep tank
tankzhang:x:0:0:root:/root:/bin/bash


例b,這個(gè)例子,用tankzhang把文件test中的root全部替換掉,請(qǐng)注意g這個(gè)字母,global的縮寫

代碼如下:


[zhangy@BlackGhost mytest]# sed 's/root/tankzhang/g' test |grep zhang
tankzhang:x:0:0:tankzhang:/tankzhang:/bin/bash
zhangy:x:1000:100:,,,:/home/zhangy:/bin/bash
ba:x:1002:1002::/home/zhangy:/bin/bash
@zhangying:*:1004:1004::/home/test:/bin/bash


例c,加了-n p后表示只打印那些發(fā)生替換的行(部分替換),上面的例子,我并沒有加上grep

代碼如下:


[zhangy@BlackGhost mytest]# sed -n 's/root/tankzhang/p' test
tankzhang:x:0:0:root:/root:/bin/bash


例d,加了-n pg后表示只打印那些發(fā)生替換的行(全部替換),上面的例子,我并沒有加上grep

代碼如下:


[zhangy@BlackGhost mytest]# sed -n 's/root/tankzhang/pg' test
tankzhang:x:0:0:tankzhang:/tankzhang:/bin/bash


例e,在第二行,到第八行之間,替換以zhang開頭的行,用ying來替換,并顯示替換的行

代碼如下:


[zhangy@BlackGhost mytest]# cat test | sed -ne '2,8s/^zhang/ying/gp'
yingy:x:1000:100:,,,:/home/zhangy:/bin/bash


例f,當(dāng)有多個(gè)命令要執(zhí)行時(shí),可以用分號(hào)來分開,并且分隔符可以自定義,默認(rèn)是/。上面的例子意思是在第二行,到第八行之間,替換以zhang開頭的行,用ying來替換,在5,到10間,用goodbay來替換dbus,并顯示替換的行

代碼如下:


[zhangy@BlackGhost mytest]# cat test | sed -n  '2,8s/^zhang/ying/gp;5,10s#dbus#goodbay#gp'
yingy:x:1000:100:,,,:/home/zhangy:/bin/bash
goodbay:x:81:81:System message bus:/:/bin/false


例g,這個(gè)例子根上面的那個(gè)例子一樣,只不過有一點(diǎn)不同,那就是-e來充當(dāng)了分號(hào)的作用,-e也能分割多個(gè)命令。

代碼如下:


[zhangy@BlackGhost mytest]# cat test | sed -ne '2,8s/zhang/ying/gp' -ne  '5,10s#dbus#goodbay#gp'
yingy:x:1000:100:,,,:/home/yingy:/bin/bash
goodbay:x:81:81:System message bus:/:/bin/false


例h,正則的用法,在sed里面用括號(hào)的話要加上\的,不然會(huì)報(bào)錯(cuò)的。

代碼如下:


[zhangy@BlackGhost mytest]# sed -ne '2,8s/^\(zhangy\)/\1ing/gp' test
zhangying:x:1000:100:,,,:/home/zhangy:/bin/bash
[root@masters ~]# sed -ne '2,8s/^\(zhangy\)/&ing/gp' test
zhangying:x:1000:100:,,,:/home/zhangy:/bin/bash


例i,&的用處是,在找到的字符串后加上&后面的字符串,zhang后都加上了ying

代碼如下:


[zhangy@BlackGhost mytest]# sed -ne '2,15s/zhang/&ying/gp' test
zhangyingy:x:1000:100:,,,:/home/zhangyingy:/bin/bash
ba:x:1002:1002::/home/zhangyingy:/bin/bash
@zhangyingying:*:1004:1004::/home/test:/bin/bash


例j,這個(gè)例子是說,在以zhang開頭的行開始,到匹配Po的行結(jié)束,在他們之間進(jìn)行替換

代碼如下:


[zhangy@BlackGhost mytest]# sed -ne '/^zhang/,/Po/s/zhang/ying/gp' test
yingy:x:1000:100:,,,:/home/yingy:/bin/bash
ba:x:1002:1002::/home/yingy:/bin/bash
@yingying:*:1004:1004::/home/test:/bin/bash


例k,n;這里的n是next的縮寫,找到root的行后,將其下一行的中的bin換成tank

代碼如下:


[zhangy@BlackGhost mytest]$ sed  '/root/{n;s/bin/tank/}' test
root:x:0:0:root:/root:/bin/bash
tank:x:1:1:bin:/bin:/bin/false


例m,y的作用是將匹配的字符換成大寫,不過替換字符和被替換字符長(zhǎng)度要一樣

代碼如下:


[zhangy@BlackGhost mytest]$ sed -e '1,2y/root/ROOT/' test
ROOT:x:0:0:ROOT:/ROOT:/bin/bash
bin:x:1:1:bin:/bin:/bin/false


例n,h的作用是將找到的行,放到一個(gè)緩存區(qū),G的作用是將緩存區(qū)中的內(nèi)容放到最后一行

代碼如下:


[zhangy@BlackGhost mytest]$ sed -e '/root/h' -e '$G' test
................................
.............................
ba:x:1002:1002::/home/zhangy:/bin/bash
test:x:1003:1003::/home/test:/bin/bash
@zhangying:*:1004:1004::/home/test:/bin/bash
root:x:0:0:root:/root:/bin/bash


例o,行替換,用匹配root的行,來替換匹配zhangy的行

代碼如下:


[zhangy@BlackGhost mytest]$ sed -e '/root/h' -e '/zhangy/g' test
root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/bin/false
daemon:x:2:2:daemon:/sbin:/bin/false
mail:x:8:12:mail:/var/spool/mail:/bin/false
ftp:x:14:11:ftp:/home/ftp:/bin/false
&nobody:$:99:99:nobody:/:/bin/false
root:x:0:0:root:/root:/bin/bash
http:x:33:33::/srv/http:/bin/false
dbus:x:81:81:System message bus:/:/bin/false
hal:x:82:82:HAL daemon:/:/bin/false
mysql:x:89:89::/var/lib/mysql:/bin/false
aaa:x:1001:1001::/home/aaa:/bin/bash
root:x:0:0:root:/root:/bin/bash
test:x:1003:1003::/home/test:/bin/bash
root:x:0:0:root:/root:/bin/bash


例p,這個(gè)例子是說,在以zhang開頭的行開始,到匹配Po的行結(jié)束,在他們之間進(jìn)行替換

代碼如下:


[zhangy@BlackGhost mytest]# sed -ne '/^zhang/,/Po/s/zhang/ying/gp' test
yingy:x:1000:100:,,,:/home/yingy:/bin/bash
ba:x:1002:1002::/home/yingy:/bin/bash
@yingying:*:1004:1004::/home/test:/bin/bash


例q,3q的意思是到第三行的時(shí)候,退出

代碼如下:


[zhangy@BlackGhost mytest]$ sed -e 's/bin/tank/g;3q' test
root:x:0:0:root:/root:/tank/bash
tank:x:1:1:tank:/tank:/tank/false
daemon:x:2:2:daemon:/stank:/tank/false


例r,特殊匹配

匹配數(shù)字別忘了中括號(hào)外面還有一個(gè)中括號(hào)。
[:alnum:] 字母數(shù)字 [a-z A-Z 0-9]
[:alpha:] 字母 [a-z A-Z]
[:blank:] 空格或制表鍵
[:cntrl:] 任何控制字符
[:digit:] 數(shù)字 [0-9]
[:graph:] 任何可視字符(無空格)
[:lower:] 小寫 [a-z]
[:print:] 非控制字符
[:punct:] 標(biāo)點(diǎn)字符
[:space:] 空格
[:upper:] 大寫 [A-Z]
[:xdigit:] 十六進(jìn)制數(shù)字 [0-9 a-f A-F]

代碼如下:


[zhangy@BlackGhost mytest]# sed -ne '2,15s/zhangy.*[[:digit:]]/=======/gp'  test
=======:,,,:/home/zhangy:/bin/bash
@=======::/home/test:/bin/bash


例2
例a,刪除1,14行

代碼如下:


[zhangy@BlackGhost test]$ sed -e '1,14d' test
@zhangying:*:1004:1004::/home/test:/bin/bash
policykit:x:102:1005:Po


例b,刪除4以后的行,包括第4行,把$當(dāng)成最大行數(shù)就行了。

代碼如下:


[zhangy@BlackGhost mytest]$ sed -e '4,$d' test
root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/bin/false
daemon:x:2:2:daemon:/sbin:/bin/false


例c,刪除包括false的行,或者包括bash的行,別忘了加\

代碼如下:


[zhangy@BlackGhost mytest]$ sed -e '/\(false\|bash\)$/d' test
policykit:x:102:1005:Po


例d,刪除從匹配root的行,到匹配以test開頭的行,中間的行

代碼如下:


[zhangy@BlackGhost mytest]$ sed -e '/root/,/^test/d' test
@zhangying:*:1004:1004::/home/test:/bin/bash
policykit:x:102:1005:Po

例3
例a,讀取test2的內(nèi)容,并將其寫入到匹配行的下面

代碼如下:


[zhangy@BlackGhost mytest]$ sed -e '/^root/r test2' test
root:x:0:0:root:/root:/bin/bash
=============
-------------
+++++++++++++
bin:x:1:1:bin:/bin:/bin/false
daemon:x:2:2:daemon:/sbin:/bin/false


例b,將匹配數(shù)字的行,寫入test2中

代碼如下:


[zhangy@BlackGhost mytest]$ sed '/[[:digit:]]/w test2' test


例c,將要插入的東西,插入匹配行的下面

代碼如下:


[zhangy@BlackGhost mytest]$ sed  '/root/a\\ ===aaaa====' test
root:x:0:0:root:/root:/bin/bash
===aaaa====
bin:x:1:1:bin:/bin:/bin/false


例d,正好根a相反,將要插入的東西,插入到匹配行的上面

代碼如下:


[zhangy@BlackGhost mytest]$ sed '/^daemon/i\\=================' test
root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/bin/false
=================
daemon:x:2:2:daemon:/sbin:/bin/false
mail:x:8:12:mail:/var/spool/mail:/bin/false

例4
#取得一個(gè)文件(或目錄)路徑的父目錄,s@@@為替換格式,\(/.*/\)是指一個(gè)"/"后面跟了任意字符又跟了一個(gè)"/",其中\(zhòng)(\)是用來把匹配內(nèi)容作為一個(gè)整體后向引用,[^/]\{1,\}是指一個(gè)非"/"字符出現(xiàn)了一次,兩次,或多次;/\?是指"/"出現(xiàn)了0次或1次,\1是后向引用前面匹配的內(nèi)容

代碼如下:


[root@practice ~]# echo "/usr/local/bin/" |sed 's@\(/.*/\)[^/]\{1,\}/\?@\1@'
/usr/local/
#使用擴(kuò)展正則表達(dá)式后,亦可如此:
[root@practice ~]# echo "/etc/rc.d/rc.sysinit" | sed -r 's@(/.*/)[^/]+/?@\1@'
/etc/rc.d/

看完上述內(nèi)容,你們掌握Linux中如何使用sed文本處理命令的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI