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=https://www.cnblogs.com/jiagooushi/archive/2022/12/19/${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 腳本檔案名和行號,
本文由
傳智教育博學谷教研團隊發布,如果本文對您有幫助,歡迎
關注和點贊;如果您有任何建議也可留言評論或私信,您的支持是我堅持創作的動力,轉載請注明出處!
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/540273.html
標籤:其他
上一篇:分布式ID生成方案
