作者 | 季文康
前言
其實,你現在閱讀到的是第三版的文章(幾乎全部重構)。記得我寫第一版的時候,還是一名 “參賽選手”。後來比賽失利便和朋友一起做 IDC 創業。第二次改的時候,是我發現閱讀量在倆三個月內直接自己站點 top 到第一,加上參與了開源社群,維護了 LCTT-CLI 專案。最後第三次也就是這次,是因為透過了 RHCE 模擬考加上一年多積累。所以這次的內容或是排版都應是最棒!BTW:這篇文章在我的部落格IT兄弟盟[2],依舊是第一的閱讀量!
一開始寫這篇文章是因為兼職創業 IDC 公司運維,需要一點 shell script 來實現某些需求。雖然現在已經是 Python 的時代了。插個話題,我怎麼理解 Python 和 Shell 呢?拿游泳來做個比喻:前者是正規游泳館,有正規教練輔助相伴;後者是鄉下小湖泊,麻雀雖小五臟俱全。人工智慧選中的 Python 勢必鋒不可當,經典的 Shell 也相當精妙絕倫。比如以前我寫 Shell 的時候用了很多 if else
陳述句、 case
陳述句,有 test
陳述句,懂得 ||
&&
;
輔助,這是最小白的。後面學習了很多比較運運算元,但多數還是在積累命令數量以及條件陳述句。再到現在,我開始去思考命令和命令之間存在的關係、陳述句分隔符的意義、BASH 控制結構等等。
所以今天和大家分享的主要是 “命令替換” 以及 “引數擴充套件” 。
什麼是命令替換
簡單的來說就是在 SHELL 內巢狀多條命令,一次性執行得到結果。
1、一層 SHELL 巢狀
# echo `whoami`
# echo $(whoami)
# echo "hello,`whoami`"
# echo "hello,$(whoami)"
2、二層 SHELL 巢狀
# echo `cat ./gn2.txt` | sed -s "s;$; --list;"
使用 ``
讀取檔案內容,再使用管道符二次處理後。執行!
註意:這裡已經用了一層巢狀,以下多個小節會套用以實現二層巢狀。
a. 使用 "$()" 進行二層巢狀
b. 使用 "|" 進行命令導向
c. 註意事項以及解答一些疑問
可能有讀者已經註意到了,之前在簡單 SHELL 一層巢狀中說了巢狀還有另一種。那為什麼不使用 ` `
進行巢狀。
` `
不支援命令巢狀執行!
~
的反引號鍵!$(...)
格式受到POSIX標準支援,也利於巢狀。
$()
可以多層巢狀類似 $($($()))
,但如果內部有一個` `
也是可以執行的喲(出於相容考慮)!3、進階
之前我們介紹了
``
和$(...)
,這倆種命令執行。想來現在你一定對命令執行有比較深的理解了。現在,我們需要再進階一下~
1、 (cmd)
與 {var}
關於 ( )
與 { }
,和 命令替換
一樣都是 shell 擴充套件
父類下的相關概念。
提示:{}
頭部大括號右側必須有一個空格,尾部括號左側必須有分號結尾。
# ( echo firest;echo second; )
# { echo third;echo fourth; }
註意: ( )
只是對一串命令重新開一個 子 shell 進行執行, { }
對一串命令在 當前 shell 執行。
2、()
與 {}
造成的影響
a. ()
括號內的陳述句影響在括號內
# var=source
# ( echo $var;var=global;echo $var; )
# echo $var
b. {}
括號內的陳述句影響到全域性。
# echo $var
# { echo $var;var=global;echo $var; }
# echo $var
註意:{}
改變 var
的變數以後,外部也受到了影響。
什麼是引數擴充套件
引數擴充套件的基本格式是 ${ parameter }
,擴充套件的結果是 ${ parameter }
被替換為相應的值。
1、實體一
echo $1 $11
echo $1 ${11}
首先解釋下 ${1..9}
是什麼意思。在我們寫 Shell
時必不可免的需要傳遞引數以實現自定義變數。當超過阿拉伯數字 9
以後。就需要使用 ${ parameter }
明確告訴Shell
第 11
個引數是 ${11}
。
提示:上圖顯示 101
就是因為 $11
不滿足 [1-9]{1}
。系統將 11
拆分成 $1
和 1
,所以運算後結果是 101
。
2、實體二
ban=ban
echo a $banana
echo a ${ban}ana
這個實體中,我想輸出 banana
。已經定義了一個 ban
的變數為 ban
,只要加上 ana
就可以成為 “笨啦啦”。
但是很顯然的不加 {}
是無法做到使變數 $ban
配合 ana
顯示出 banana
的!
什麼是變數擴充套件
從官方定義上來說,我並不應該將 “變數擴充套件” 無中生有出來。
"
$
字元引入引數擴充套件,命令替換或算術擴充套件。" —— 官方手冊
主要是出於倆個方面考慮:
${...}
括號內的東西,而變數一詞可表示所有操作圍繞變數展開。實體:
var='This is one test sentence.'
var1=parameter
var2=word
現在我們有了這樣的一個句子,我希望做一些判斷、摘取(或者說:切片)或修改。我該如何操作?
1、變數替換
a. ${parameter:-word}
# echo ${var1:-$var2}
parameter
# var1=
# echo ${var1:-$var2}
word
如果 var1
未設定或為空,則替換成 var2
。
b. ${parameter:=word}
同上。位置引數和特殊引數不能以這種方式分配。
c. ${parameter:?word}
# var1=
# echo ${var1:?var2}
bash: var1: var2
# echo $?
1
當變數 var1
未設定或為空,shell 也是可互動時,進行報錯並且退出。如果 shell 不可互動,則發生變數替換。
d. ${parameter:+word}
# echo $var1
parameter
# echo $var2
word
# echo ${var1:+$var2}
word
# echo $var1
parameter
如果 var1
為空或未設定,那麼就什麼都不做。不然使用 var2
進行替換。
提示: 在我測試的時候,我發現並不是全域性生效的。
2、變數切片
a. 範圍切片(同方向)
# echo ${var:8:17}
one test sentence
註意:倆個數字都是從頭開始數的。
b. 範圍切片(非同向)
# echo ${var:8:-1}
one test sentence
# echo ${var:8:(-1)}
one test sentence
提示:倆種寫法都是正確的。
c. 切片位置
# a='This is one'
# echo ${#a}
11
提示 :首先建立變數 a='This is one'
,然後使用 echo ${#a}
將字元數量讀了出來。
3、變數修改
a. 簡單修改
# echo ${var}
This is one test sentence.
# echo ${var/one/a}
This is a test sentence.
提示:個人認為這種是最好的方式了,可以範圍式修改(包含刪除)。
b. 簡單刪除
# echo ${var%sentence.}
This is one test
# echo ${var#This is}
one test sentence.
c. 附:表格
變數設定方式 | 說明 |
---|---|
${變數#關鍵字} | 若變數內容從頭開始的資料符合“關鍵字”,則將符合的最短資料刪除 |
${變數##關鍵字} | 若變數內容從頭開始的資料符合“關鍵字”,則將符合的最長資料刪除 |
${變數%關鍵字} | 若變數內容從尾開始的資料符合“關鍵字”,則將符合的最短資料刪除 |
${變數%%關鍵字} | 若變數內容從尾開始的資料符合“關鍵字”,則將符合的最長資料刪除 |
${變數/舊字串/新字串} | 若變數內容符合“舊字串”,則首個舊字元會被新字元替換。 |
${變數/舊字串//新字串} | 若變數內容符合“舊字串”,則全部舊字元會被新字元替換。 |
更深入學習,探索資料