Shell 變數知多少?
Shell 變數(一)
bash shell 程式設計和其他程式語言差不多,同樣包含變數(存放字串和數值的容器,可以進行修改、比較、傳遞)。在引用 bash 變數時,可以使用一些非常特殊的運算子。bash 還擁有內建變數,這些變數可以提供有關指令碼中其他變數的重要資訊。下面介紹 bash 變數和一些特殊的變數引用機制,展示如何將其運用於你自己的指令碼。
1、shell 變數基礎知識
bash 指令碼中的變數名稱通常採用全大寫,但這並非強制性的,只是一種常見做法而已。變數不用事先宣告,直接使用就行了。變數基本上都是字串型別,不過有些運算子能夠將變數內容視為數字。變數的實際用法如下所示。
# 使用shell變數的普通指令碼
MYVAR="something"
echo $MYVAR
# 寫法類似,但沒有引號
MY_2ND=anotherone
echo $MY_2ND
# 這裡因為包含空客,需要使用引號:
MYOTHER="more stuff to echo"
echo $MYOTHER
bash 變數的語法有兩處要點,但可能不那麼一目瞭然。
- 首先,賦值語法 name=value 看起來相當直觀,但 = 兩側不能有任何空白字元。如果允許 = 兩側出現空白字元,那麼變數賦值就會變成下面這樣:
MYVAR = something
此時 shell 很難區分出到底是要呼叫命令還是要給變數賦值。對於能夠以 = 為引數的命令(如 test)更是如此。所以,還是讓事情簡單點吧:變數賦值時,shell 不允許在 = 兩側出現空白字元。該規定的另一方面也值得注意,不要在檔名中使用 =。
-
其次需要注意的是,引用變數時要使用 $ 符號。給變數賦值時不需要在變數名前加 $,但獲取變數值時需要。出現在表示式 $(( ))中的變數是個例外。原因很簡單,就是為了消除歧義。如下:
MYVAR=something echo MYVAR is now MYVAR
你能分辨出哪個是字串 MYVAR,哪個是變數 MYVAR 嗎?bash 中的一切都是字串,所以需要用 $ 來表明變數引
用。
2、記錄指令碼
詳細討論 shell 指令碼或變數前,我們還得說說如何記錄指令碼。畢竟,你得能看明白自己的指令碼,即便是在編寫完的幾個月後。
用註釋記錄指令碼。# 代表註釋的開始。該行上隨後的所有字元都會被shell 忽略。
#
# 這是一行註釋
#
# 多用註釋
# 註釋是你的好朋友
如果您是java開發工作者,你會發現,這就是我們平時常說的程式碼註釋。
3、將變數名與周圍的文字分開
如果你需要輸出變數以及其他文字。引用變數要用到 $,但是該怎麼區分變數名與緊隨其後的其他文字呢?例如,你想要用 shell 變數作為檔名的一部分,如下所示:
for FN in 1 2 3 4 5
do
somescript /tmp/rep$FNport.txt #執行某個指令碼,把檔案當作執行引數,如cat
done
shell 會怎麼理解這段程式碼?它會認為變數名從 $ 開始,到點號結束。換句話說,它將 $FNport 視為變數名,而非我們想要的 $FN。
那麼,我們如何讓shell知道我們的變數是FN呢?
使用完整的變數引用語法,不僅要包括 $,還要在變數名周圍加上花括號,如下:
somescript /tmp/rep${FN}port.txt
因為 shell 變數名中只能包含字母、數字以及下劃線,所以很多時候並不需要使用花括號。任何空白字元或標點符號(下劃線除外)都足以提示變數名的結束位置。但只要有疑問,就應該用花括號。
4、匯出變數
你在某個指令碼中定義了一個變數,但在呼叫其他指令碼時,該指令碼並不知道這個變數的存在。為了解決這個問題,我們需要將傳給其他指令碼的變數匯出。如下所示:
export MYVAR
export NAME=value
要想檢視所有已匯出的變數,敲入命令 env(或者內建命令 export-p)就能列出各個變數及其值。當指令碼執行時,這些變數都可供使用,其中很多是 bash 啟動指令碼已經設定好的,如$PATH。
可以在 export 後面跟上變數賦值,不過這種寫法不適用於比較老的 shell 版本。然後匯出之後,就可以隨意給變數賦值,不用重複匯出。因此,有時你會看到下列語句:
# 匯出變數
export FNAME
export SIZE
export MAX
# 為變數賦值
MAX=2048
SIZE=64
FNAME=/tmp/scratch
注意,匯出的變數實際上是按值呼叫的。在被呼叫指令碼中修改匯出變數的值並不會改變呼叫指令碼中該變數的值。
對於匯出的變數,我們該如何刪除呢?
# 刪除變數
unset myvar
Shell 變數(二)
你希望使用者能在呼叫指令碼時指定引數。可以要求使用者設定一個 shell變數,但這種做法似乎不夠靈活。另外還需要向其他指令碼傳遞資料。這可以通過環境變數實現,但會使兩個指令碼之間的聯絡過於緊密。因此,此處我們可以用到指令碼引數。
1、在shell指令碼中使用引數
使用命令列引數。在命令列上,出現在指令碼名之後的任意單詞都可以在指令碼中作為編號變數(numbered variable)被訪問。假設有下列指令碼 simplest.sh。
# 一個簡單的shell指令碼
echo $1
該指令碼會顯示在命令列上被呼叫時所指定的第一個引數。我們來看一種實際用法。
$ cat simplest.sh
# 一個簡單的shell指令碼
echo ${1}
$ ./simplest.sh you see what I mean
you
$ ./simplest.sh one more time
one
$
其他引數的可用形式分別為 ${2}、${3}、${4}、${5} 等。單個數位的數字用不著花括號,除非要區分變數名與其後出現的文字。典型的指令碼只用到少部分引數,但如果涉及 ${10},那就得使用花括號了,否則 shell 會將 $10 理解為 ${1} 後面緊跟著字串 0,如下所示。
$ cat tricky.sh
echo $1 $10 ${10}
$ ./tricky.sh I II III IV V VI VII VIII IX X XI
I I0 X #注意觀察第二個輸出
$
第 10 個引數的值是 X,但如果在指令碼中寫成 $10,那麼你在 echo語句中得到的會是第一個引數 $1,後面緊跟著一個字串 0。
因為第三個使用了${},所以三個${10}可以正常輸出X。
2、遍歷傳入指令碼的引數: $*
如果你想對指定的一系列引數執行某些操作。在編寫 shell 指令碼時,對單個引數進行處理不是什麼問題,只需要用 $1 引用這個引數即可。但如果面對的是一大批檔案呢?你可能想這樣呼叫指令碼。
./actall *.txt
shell 會進行模式匹配,生成匹配 *.txt 模式(以 .txt 結尾的檔名)的檔名列表。對於指令碼而言,我們永遠無法預估傳入的引數的個數,那麼我們就無法通過${數字}獲取所有引數,那麼${數字}方式將不再適用。
特殊的 shell 變數 $* 能夠引用所有的引數,可以將其用於 for 迴圈,如下所示:
#!/usr/bin/env bash
# 例項檔案:actall.sh
# 批量修改檔案許可權
#
for FN in $*
do
echo changing $FN
chmod 0750 $FN
done
變數 $FN 是我們自己挑選的;使用別的變數名也沒有任何問題。$*引用的是命令列上出現的所有引數。假如使用者輸入
./actall abc.txt another.txt allmynotes.txt
呼叫該指令碼時,$1 等於 abc.txt、$2 等於 another.txt、$3 等於allmynotes.txt,而 $* 等於整個引數列表。換句話說,shell 替換for 語句中的 $* 後,指令碼就變成了如下這樣:
for FN in abc.txt another.txt allmynotes.txt
do
echo changing $FN
chmod 0750 $FN
done
for 迴圈從列表中獲取第一個值,並將其賦給變數 $FN,然後執行do 和 done 之間的語句。列表中的其他值會重複執行該過程。
3、處理包含空格的引數: “”
你編寫了一個可以接受檔名作為引數的指令碼,看起來一切正常,但有一次指令碼出現了問題,結果發現是因為檔名中帶有空格。你得仔細將所有可能包含檔名的命令引數全部加上引號。引用變數時,將其放入雙引號中。
在 shell 指令碼中,曾經簡單的寫作 ls -l $1 的地方,現在最好給引數加上引號,改寫成 ls -l "$1"。否則,如果引數包含空格,那麼會被 shell 解析成兩個單詞,$1 中只會包含部分檔名。如下:
$ cat simpls.sh
# 一個簡單的shell指令碼
ls -l ${1}
$
$ ./simple.sh Oh the Waste
ls: Oh: No such file or directory
$
如果呼叫指令碼時沒有將檔名放進引號,那麼 bash 會看到 3 個引數並將 $1 替換成第 1 個引數(Oh)。ls 命令執行時只有一個引數Oh,結果就是無法找到該檔案。
接下來,我們在呼叫指令碼時給檔名加上引號。
$ ./simpls.sh "Oh the Waste"
ls: Oh: No such file or directory
ls: the: No such file or directory
ls: Waste: No such file or directory
$
還是不行。bash 得到了一個包含 3 個單詞的檔名,並將 ls 命令中的 $1 替換成了該檔名。到目前一切都還好。但是,我們並沒有將指令碼中的變數引用放入引號,因此 ls 將檔名中的各個單詞視為單獨的引數(作為單獨的檔名)。結果還是無法找到這些檔案,相當執行命令:
ls -l Oh the Waste
因此,我們需要將我們變數引用放進引號,修改指令碼內容如下:
$ cat simpls.sh
# 一個簡單的shell指令碼,注意此處${1}與第一次腳本里的區別,多了雙引號
ls -l "${1}"
$
$ ./simple.sh "Oh the Waste"
$
4、處理包含空格的引數列表:$@
對於第二節,我們通過$*,可以獲取引數列表,那麼如果這個時候我們傳入的引數列表包含空格會不會有問題呢? 如下所示:
$ ./actall.sh "Oh the Waste"
changing Oh
chmod: 無 法 訪 問 "Oh": 沒 有 那 個 文 件 或 目 錄
changing the
chmod: 無 法 訪 問 "the": 沒 有 那 個 文 件 或 目 錄
changing Waste
chmod: 無 法 訪 問 "Waste": 沒 有 那 個 文 件 或 目 錄
$
按照上節中的建議,你給變數加上引號,但是仍然出現了錯誤。如下:
#!/usr/bin/env bash
#例項檔案:actall.sh
#批量修改檔案許可權
#
for FN in $*
do
echo changing "$FN"
chmod 0750 "$FN"
done
如果檔名中帶有空格,就會報錯,報錯的原因與 for 迴圈中使用的 $* 有關。在這個示例中,我們需要用到另一個不同但相關的 shell 變數 $@。如果該變量出現在引號中,則會得到一個命令列引數列表,其中每個引數都會被單獨引用起來。修改後的 shell 指令碼如下:
#!/usr/bin/env bash
# 例項檔案:chmod_all.2
# 在檔名包含空格時選擇更好的引號新增方式,批量修改檔案許可權
#
for FN in "$@"
do
chmod 0750 "$FN"
done
如果不加引號,$* 和 $@ 沒什麼兩樣。但當兩者出現在引號中時,bash 就會區別對待了。"$*" 得到的是整個引數列表,而"$@" 得到的可不是一個字串,而是與各個引數對應的帶有引號的字串列表。
如果你知道檔名中沒有空格,沿用老的 $ 語法基本沒什麼大礙。對於那些更穩健的指令碼而言,安全起見,建議使用 "$@"*
Shell 變數(三)
1、統計引數數量
你想知道呼叫指令碼時使用了多少個引數。使用 shell 內建變數 $#。如下,展示了一個嚴格要求3個引數的指令碼:
#!/usr/bin/env bash
# 例項檔案:check_arg_count
#
# 檢查正確的引數數量:
# 使用下列語法或者:if [ $# -lt 3 ]
if (( $# < 3 ))
then
printf "%b" "Error. Not enough arguments.\n" >&2
printf "%b" "usage: myscript file1 op file2\n" >&2
exit 1
elif (( $# > 3 ))
then
printf "%b" "Error. Too many arguments.\n" >&2
printf "%b" "usage: myscript file1 op file2\n" >&2
exit 2
else
printf "%b" "Argument count correct. Proceeding...\n"
fi
以下分別是引數過多和引數數量正好時的執行情況。
$ ./myscript myfile is copied into yourfile
Error. Too many arguments.
usage: myscript file1 op file2
$ ./myscript myfile copy yourfile
Argument count correct. Proceeding...
我們用 if 測試所提供的引數數量(儲存在 $# 中)是否大於 3。如果答案是肯定的,則輸出一條錯誤資訊,提醒使用者正確的指令碼用法,然後退出。
標準提示錯誤資訊會被重定向到標準錯誤(>&2)。這種做法符合標準錯誤的本意:作為所有錯誤資訊的通道。
2、丟棄引數:shift
所有的正式指令碼可能都要有兩種引數:修改指令碼行為的選項以及要處理的真正引數。你需要用一種方法在處理完選項後將其丟棄。例如,現在有如下指令碼:
for FN in "$@"
do
echo changing $FN
chmod 0750 "$FN"
done
指令碼內容非常簡單,它會顯示正在處理的檔名,然後修改檔案許可權。但有時你希望指令碼靜悄悄地工作,不要顯示檔名,而有時又希望顯示檔名。如何在保留for 迴圈的同時新增一個能夠關閉檔名顯示的選項呢?
用 shift 刪除處理過的引數,如下:
# 自定義變數
VERBOSE=0
# 判斷第一個引數的值
if [[ $1 = -v ]]
then
# 使用變數儲存引數值
VERBOSE=1
# 刪除引數
shift
fi
# 此時for拿到的引數已經少了$1,從$2開始讀取
for FN in "$@"
do
if (( VERBOSE == 1 ))
then
echo changing $FN
fi
chmod 0750 "$FN"
done
我們添加了標記變數 $VERBOSE,藉此瞭解是否應該輸出所處理的檔名。可是一旦 shell 指令碼發現 -v 選項並設定好標記,我們就用不著引數列表中的 -v 了。shift 語句告訴 bash 將命令列引數挪動一個位置,使 $2 變成 $1、$3 變成 $2,以此類推,這樣就丟棄了第一個引數($1)。如此一來,當 for 迴圈啟動時,引數列表($@)中就再也沒有 -v,剩下的是緊隨其後的那些引數。
執行結果如下:
# 執行指令碼,帶-v引數,輸出修改的檔名
$ ./shift_test.sh -v error.out
changing error.out
# 執行指令碼,不帶-v引數,悄悄執行,不輸出檔名
$./shift_test.sh error.out
$
3、獲取預設值:${:-}
有一個可以接受命令列引數的 shell 指令碼。如你希望能夠提供預設值,這樣就不用每次都讓使用者輸入那些頻繁用到的值了。
用 ${:-} 語法引用引數並提供預設值,如下所示:
FILEDIR=${1:-/tmp}
在引用 shell 變數時,有多種特殊運算子可用。:- 運算子的意思是,如果指定引數(這裡是 $1)不存在或為空,則將運算子之後的內容(本例為 /tmp)作為值。否則,使用已經設定好的引數值。該運算子可用於任何 shell 變數,並不侷限於位置引數$1、$2、$3等。
當然,你也可以用更多的程式碼來實現:用 if 語句檢查變數是否為空或不存在,但在 shell 指令碼中,此類處理司空見慣,:- 運算子可謂是一種頗受歡迎的便捷寫法。
4、設定環境變數預設值: ${HOME:=/tmp}
你的指令碼依賴於某些常用(如 $USER)或業務特定的環境變數。要想構建一個穩健的 shell 指令碼,就得確保這些變數都有合理的預設值。那麼該如何確保呢?
首次引用 shell 變數時,如果該變數沒有值,則使用賦值運算子為其賦值,如下:
cd ${HOME:=/tmp}
示例中所引用的 $HOME 會返回其當前值,除非它為空或者壓根就沒設定。對於後兩種情況(為空或沒有設定),返回 /tmp,該值還會被賦給 $HOME,隨後再引用 $HOME 的話,返回的就是這個新值。如下所示,
注意:下面的例子會改變環境變數HOME,請慎重執行
$ echo ${HOME:=/tmp}
/home/uid002
$ unset HOME # 刪除環境變數值
$ echo ${HOME:=/tmp} # 重新獲取,此時不存在,將重新賦值並返回新值
/tmp
$ echo $HOME # 此時再檢視變數,輸出新設定的值
/tmp
$ cd;pwd
/tmp
$
賦值運算子有一個重要的例外:不能對位置引數(如 $1 或$*)賦值。在這種情況下,可以使用 :-(如 ${1:-default}),該表示式只返回值,但不進行賦值。
${VAR:=value} 和 ${VAR:-value} 在形式上的差異,也許可以幫助你記憶這兩種讓人抓狂的符號。:= 執行賦值操作,同時返回運算子右側的值。:- 只做了前者一半的工作:返回值,但不賦值。因此,它的符號也只有等號的一半(一個橫槓,而不是兩個)。
Shell 變數(四)
1、獲得某個數的絕對值
變數中的數值可能是負數,也可能是零或正數。你想得到它的大小(也就是絕對值),但 bash 似乎沒有求絕對值的功能。但是,我們可以利用字串操作。如下:
${MYVAR#-}
這是一種簡單的字串操作。# 從字串起始位置開始搜尋負號(-)。如果找到,則將其刪除;如果沒有找到,就保留原值。不管是哪種情況,最後得到的都是不包含負號的絕對值。
然而,我們也可以使用 if/then/else 按照數學方法來實現。如下:
# 通過判斷數值於0的關係,並且通過與-1相乘
if (( MYVAR < 0 ))
then
let MYVAR=MYVAR*-1
fi
對比上面2種方法,明顯第一種更簡單,所以推薦第一種。
2、選取CSV的替換值
你想製作一個由逗號分隔的值列表,但不希望開頭或結尾處出現逗號,然後這是我們日常工作中很普遍的需求。如果在迴圈內部用 LIST="${LIST},${NEWVAL}" 構建該列表,那麼第一次迴圈(此時 LIST 為空)過後會得到一個前導逗號。你可以對 LIST 進行特殊的初始化處理,以便它在進入迴圈前就先得到第一個值,但如果覺得這種方法不實用,或是為了避免重複程式碼(用於得到新值),你可以改用 bash 中的 ${:+} 語法。如下:
LIST="${LIST}${LIST:+,}${NEWVAL}"
如果 {LIST} 為空或不存在,那麼 $LIST 的兩個表示式不會產生任何內容。這就意味著,第一次迴圈過後,LIST 中儲存的只有NEWVAL 的值。如果 LIST 不為空,則第二個表示式 ${LIST:+,}會被替換為逗號,將先前的值與新值分隔開來。
下面的程式碼片段用於讀取並構建 CSV 列表。
LIST=""
for NEWVAL in "$@"
do
LIST="${LIST}${LIST:+,}${NEWVAL}"
done
echo ${LIST}
3、使用陣列變數
到目前為止,我們已經見識了不少使用變數的指令碼,但是 bash 能不能處理陣列呢?當然可以,bash 有專門的一維陣列語法。
如果編寫指令碼時已經知道具體的值,則初始化陣列就很容易了。格式如下:
MYRA=(first second third home)
注意:陣列是用(),這同java裡的陣列符號[]不同。
括號內列表的每個單詞都對應著一個數組元素。你可以像下面這樣引用各個元素:
echo runners on ${MYRA[0]} and ${MYRA[2]}
輸出結果如下:
runners on first and third
注意:如果只寫 $MYRA,那麼只會得到第一個陣列元素,相當於${MYRA[0]}。
4、轉換大小寫
bash 4.0 中的幾個運算子可以在引用變數名時轉換其大小寫。如果變數 $FN 中包含一個需要轉換成小寫的檔名(字串),那麼${FN,,} 會返回全部是小寫形式的字串。與此類似,${FN^^} 會返回全部是大寫形式的字串。甚至還有 ${FN~~},它可以切換大小寫,將所有的小寫字母轉換成大寫,大寫字母轉換成小寫。
以下的 for 迴圈會將所有引數更改成小寫字母。
for FN in "$@"
do
echo "${FN}" 轉為小寫的結果為:"${FN,,}"
done
或者寫成單行指令碼:
for FN in "$@"; do echo "${FN}" 轉為小寫的結果為:"${FN,,}" ; done
5、對不存在的引數輸出錯誤訊息
有時你需要強制使用者指定某個值,否則就無法繼續往下進行。使用者有可能會遺漏某個引數,因為他們確實不知道該怎樣呼叫指令碼。你希望能給使用者點提示,省得他們自己瞎猜。相較於堆砌成堆的 if 語句,有沒有更簡潔的方法來檢查各個引數?
引用引數時使用 ${:?} 語法。如果指定引數不存在或為空,那麼 bash 會輸出錯誤訊息並退出。
#!/usr/bin/env bash
# 例項檔案:check_unset_parms
#
USAGE="usage: myscript scratchdir sourcefile conversion"
FILEDIR=${1:?"Error. You must supply a scratch directory."}
FILESRC=${2:?"Error. You must supply a source file."}
CVTTYPE=${3:?"Error. ${USAGE}"}
如果執行指令碼時沒有指定足夠的引數,則會出現下列結果。
$ ./myscript /tmp /dev/null
./myScript.sh:行11: 3: Error. usage: myscript scratchdir sourcefile conversion
$
bash 會測試各個引數,如果引數不存在或為空,則輸出錯誤資訊並退出。$3 所對應的錯誤訊息中使用了另一個 shell 變數。
如果 $3 不存在,則錯誤訊息中會包含短語 "Error."、變數$USAGE 的值。
另一方面,${:?} 生成的錯誤資訊包含 shell 指令碼檔名和行號。
本文由
傳智教育博學谷
教研團隊釋出。如果本文對您有幫助,歡迎
關注
和點贊
;如果您有任何建議也可留言評論
或私信
,您的支援是我堅持創作的動力。轉載請註明出處!
- ElasticSearch還能效能調優,漲見識、漲見識了!!!
- 【必須收藏】別再亂找TiDB 叢集部署教程了,這篇保姆級教程來幫你!!| 博學谷狂野架構師
- 【建議收藏】7000 字的TIDB保姆級簡介,你見過嗎
- Tomcat架構設計剖析 | 博學谷狂野架構師
- 你可能不那麼知道的Tomcat生命週期管理 | 博學谷狂野架構師
- 大哥,這是併發不是並行,Are You Ok?
- 為啥要重學Tomcat?| 博學谷狂野架構師
- 這是一篇純講SQL語句優化的文章!!!| 博學谷狂野架構師
- 捲起來!!!看了這篇文章我才知道MySQL事務&MVCC到底是啥?
- 為什麼99%的程式設計師都做不好SQL優化?
- 如何搞定MySQL鎖(全域性鎖、表級鎖、行級鎖)?這篇文章告訴你答案!太TMD詳細了!!!
- 【建議收藏】超詳細的Canal入門,看這篇就夠了!!!
- 從菜鳥程式設計師到高階架構師,竟然是因為這個字final
- 為什麼95%的Java程式設計師,都是用不好Synchronized?
- 99%的Java程式設計師者,都敗給這一個字!
- 8000 字,就說一個字Volatile
- 98%的程式設計師,都沒有研究過JVM重排序和順序一致性
- 來一波騷操作,Java記憶體模型
- 時隔多年,這次我終於把動態代理的原始碼翻了個地兒朝天
- 再有人問你分散式事務,把這篇文章砸過去給他