ver:1.0 博客:https://www.cnblogs.com/Rohn 本文介紹了Shell編程的一些語法規範,主要參考依據為谷歌的Shell語法風格。 背景 博客:https://www.cnblogs.com/Rohn 使用哪一種Shell 可執行文件必須以 #!/bin/bash 和最 ...
- ver:1.0
- 博客:https://www.cnblogs.com/Rohn
- 本文介紹了Shell編程的一些語法規範,主要參考依據為谷歌的Shell語法風格。
目錄
背景
博客:https://www.cnblogs.com/Rohn
使用哪一種Shell
可執行文件必須以 #!/bin/bash
和最小數量的標誌開始。請使用 set 來設置shell的選項,使得用 <script_name>
調用你的腳本時不會破壞其功能。
推薦使用:
#!/usr/bin/env bash
env
一般固定在/usr/bin
目錄下,而其餘解釋器的安裝位置就相對不那麼固定。
限制所有的可執行Shell腳本為bash使得我們安裝在所有電腦中的shell語言保持一致性。
無論你是為什麼而編碼,對此唯一例外的是當你被迫時可以不這麼做的。其中一個例子是Solaris SVR4
包,編寫任何腳本都需要用純Bourne shell
。
[root@test ~]# echo $SHELL
/bin/bash
什麼時候使用Shell
使用Shell需要遵守的一些準則:
- 如果你主要是在調用其他的工具並且做一些相對很小數據量的操作,那麼使用Shell來完成任務是一種可接受的選擇。
- 如果你在乎性能,那麼請選擇其他工具,而不是使用Shell。
- 如果你發現你需要使用數據而不是變數賦值(如 ${PHPESTATUS} ),那麼你應該使用Python腳本。
- 如果你將要編寫的腳本會超過100行,那麼你可能應該使用Python來編寫,而不是Shell。
請記住,當腳本行數增加,儘早使用另外一種語言重寫你的腳本,以避免之後花更多的時間來重寫。
註釋
博客:https://www.cnblogs.com/Rohn
Bash只支持單行註釋,使用#
開頭的都被當作註釋語句。
頂層註釋
每個文件必須包含一個頂層註釋,對其內容進行簡要概述。版權聲明和作者信息是可選的。
例如:
#!/usr/bin/env bash
# Author: Rohn
# Version: 1.0
# Created Time: 2020/06/06
# Perform hot backups of MySQL databases.
- 第1行,指明解釋器,使用
bash
#!
叫做"Shebang"或者"Sha-bang"(Unix術語中,#
號通常稱為sharp,hash或mesh;而!
則常常稱為bang),指明瞭執行這個腳本文件的解釋程式。當然,如果使用bash test.sh
這樣的命令來執行腳本,那麼#!
這一行將會被忽略掉。
- 第2-5行,分別為作者、版本號、創建時間、功能說明。
功能註釋
任何不是既明顯又短的函數都必須被註釋。任何庫函數無論其長短和複雜性都必須被註釋。
其他人通過閱讀註釋(和幫助信息,如果有的話)就能夠學會如何使用你的程式或庫函數,而不需要閱讀代碼。
所有的函數註釋應該包含:
- 函數的描述
- 全局變數的使用和修改
- 使用的參數說明
- 返回值,而不是上一條命令運行後預設的退出狀態
例如:
#!/usr/bin/env bash
# Author: Rohn
# Version: 1.0
# Created Time: 2020/06/06
# Perform hot backups of Oracle databases.
export PATH='/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin'
#######################################
# Cleanup files from the backup dir
# Globals:
# BACKUP_DIR
# ORACLE_SID
# Arguments:
# None
# Returns:
# None
#######################################
cleanup() {
...
}
TODO註釋
TODOs應該包含全部大寫的字元串TODO
,接著是括弧中你的用戶名。冒號是可選的。最好在TODO條目之後加上bug
或者ticket
的序號。
例如:
# TODO(mrmonkey): Handle the unlikely edge cases (bug ####)
格式
博客:https://www.cnblogs.com/Rohn
縮進
縮進兩個空格,沒有製表符。例如:
if [ a > 1 ];then
echo '${a} > 1'
fi
行的長度和長字元串
行的最大長度為80個字元。例如:
# DO use 'here document's
cat <<END;
I am an exceptionally long
string.
END
# Embedded newlines are ok too
long_string="I am an exceptionally
long string."
管道
如果一行容不下整個管道操作,那麼請將整個管道操作分割成每行一個管段。
應該將整個管道操作分割成每行一個管段,管道操作的下一部分應該將管道符放在新行並且縮進2個空格。這適用於使用管道符|
的合併命令鏈以及使用||
和&&
的邏輯運算鏈。
例如:
# All fits on one line
command1 | command2
# Long commands
command1 \
| command2 \
| command3 \
| command4
迴圈
if-else語句
if
和; then
放在同一行,;
後空一格,else
單獨一行,fi
單獨一行,並與if
垂直對齊。即:
if condition; then
statement(s)
else
statement(s)
fi
for-do和while-do語句
while/for
和; do
放在同一行,done
與while/for
垂直對齊,即:
# while structure
while condition; do
statement(s)
done
# for structure
for condition; do
statement(s)
done
例如:
for dir in ${dirs_to_cleanup}; do
if [[ -d "${dir}/${ORACLE_SID}" ]]; then
log_date "Cleaning up old files in ${dir}/${ORACLE_SID}"
rm "${dir}/${ORACLE_SID}/"*
if [[ "$?" -ne 0 ]]; then
error_message
fi
else
mkdir -p "${dir}/${ORACLE_SID}"
if [[ "$?" -ne 0 ]]; then
error_message
fi
fi
done
case語句
- 通過2個空格縮進可選項。
- 在同一行可選項的模式右圓括弧之後和結束符
;;
之前各需要一個空格。 - 長可選項或者多命令可選項應該被拆分成多行,模式、操作和結束符
;;
在不同的行。
匹配表達式比case
和esac
縮進一級。多行操作要再縮進一級。一般情況下,不需要引用匹配表達式。模式表達式前面不應該出現左括弧。避免使用;&
和;;&
符號。即:
# case structure
case in expression in
pattern1)
statement1
;;
pattern2)
statement2
;;
...
*)
statementn
;;
esac
例如:
case "${expression}" in
a)
variable="..."
some_command "${variable}" "${other_expr}" ...
;;
absolute)
actions="relative"
another_command "${actions}" "${other_expr}" ...
;;
*)
error "Unexpected expression '${expression}'"
;;
esac
只要整個表達式可讀,簡單的命令可以跟模式和;;
寫在同一行。這通常適用於單字母選項的處理。當單行容不下操作時,請將模式單獨放一行,然後是操作,最後結束符;;
也單獨一行。當操作在同一行時,模式的右括弧之後和結束符;;
之前請使用一個空格分隔。
verbose='false'
aflag=''
bflag=''
files=''
while getopts 'abf:v' flag; do
case "${flag}" in
a) aflag='true' ;;
b) bflag='true' ;;
f) files="${OPTARG}" ;;
v) verbose='true' ;;
*) error "Unexpected option ${flag}" ;;
esac
done
變數擴展
按優先順序順序:保持跟你所發現的一致;引用你的變數;推薦用
${var}
而不是$var
。
例如
# Section of recommended cases.
# Preferred style for 'special' variables:
echo "Positional: $1" "$5" "$3"
echo "Specials: !=$!, -=$-, _=$_. ?=$?, #=$# *=$* @=$@ \$=$$ ..."
# Braces necessary:
echo "many parameters: ${10}"
# Braces avoiding confusion:
# Output is "a0b0c0"
set -- a b c
echo "${1}0${2}0${3}0"
# Preferred style for other variables:
echo "PATH=${PATH}, PWD=${PWD}, mine=${some_var}"
while read f; do
echo "file=${f}"
done < <(ls -l /tmp)
# Section of discouraged cases
# Unquoted vars, unbraced vars, brace-quoted single letter
# shell specials.
echo a=$avar "b=$bvar" "PID=${$}" "${1}"
# Confusing use: this is expanded as "${1}0${2}0${3}0",
# not "${10}${20}${30}
set -- a b c
echo "$10$20$30"
特性
博客:https://www.cnblogs.com/Rohn
命令替換
使用
$(command)
而不是反引號。
嵌套的反引號要求用反斜杠轉義內部的反引號。而$(command)
形式嵌套時不需要改變,而且更易於閱讀。
例如:
# This is preferred:
var="$(command "$(command1)")"
# This is not:
var="`command \`command1\``"
文件名的通配符擴展
當進行文件名的通配符擴展時,請使用明確的路徑。
因為文件名可能以-
開頭,所以使用擴展通配符./*
比*
來得安全得多。
# Here's the contents of the directory:
# -f -r somedir somefile
# This deletes almost everything in the directory by force
psa@bilby$ rm -v *
removed directory: `somedir'
removed `somefile'
# As opposed to:
psa@bilby$ rm -v ./*
removed `./-f'
removed `./-r'
rm: cannot remove `./somedir': Is a directory
removed `./somefile'
命名約定
博客:https://www.cnblogs.com/Rohn
函數名
使用小寫字母,並用下劃線分隔單詞。使用雙冒號
::
分隔庫。函數名之後必須有圓括弧。關鍵詞function
是可選的,但必須在一個項目中保持一致。
如果你正在寫單個函數,請用小寫字母來命名,並用下劃線分隔單詞。如果你正在寫一個包,使用雙冒號 ::
來分隔包名。大括弧必須和函數名位於同一行(就像在Google的其他語言一樣),並且函數名和圓括弧之間沒有空格。
# Single function
my_func() {
...
}
# Part of a package
mypackage::my_func() {
...
}
當函數名後存在 ()
時,關鍵詞 function
是多餘的。但是其促進了函數的快速辨識。
變數名
使用小寫字母,迴圈的變數名應該和迴圈的任何變數同樣命名。例如:
for zone in ${zones}; do
something_with "${zone}"
done
常量和環境變數名
全部使用大寫字母,用下劃線分隔,聲明在文件的頂部。例如:
# Constant
readonly PATH_TO_FILES='/some/path'
# Both constant and environment
declare -xr ORACLE_SID='PROD'
源文件名
使用小寫字母,如果需要的話使用下劃線分隔單詞。例如: maketemplate
或者 make_template
,而不是 make-template
。
只讀變數
使用小寫字母,使用 readonly
或者 declare -r
來確保變數只讀。
因為全局變數在Shell中廣泛使用,所以在使用它們的過程中捕獲錯誤是很重要的。當你聲明瞭一個變數,希望其只讀,那麼請明確指出。
zip_version="$(dpkg --status zip | grep Version: | cut -d ' ' -f 2)"
if [[ -z "${zip_version}" ]]; then
error_message
else
readonly zip_version
fi
使用本地變數
使用小寫字母,使用 local
聲明特定功能的變數。聲明和賦值應該在不同行。
使用 local
來聲明局部變數以確保其只在函數內部和子函數中可見。這避免了污染全局命名空間和不經意間設置可能具有函數之外重要性的變數。
當賦值的值由命令替換提供時,聲明和賦值必須分開。因為內建的 local
不會從命令替換中傳遞退出碼。
my_func2() {
local name="$1"
# Separate lines for declaration and assignment:
local my_var
my_var="$(my_func)" || return
# DO NOT do this: $? contains the exit code of 'local', not my_func
local my_var="$(my_func)"
[[ $? -eq 0 ]] || return
...
}
調用命令
博客:https://www.cnblogs.com/Rohn
檢查返回值
對於非管道命令,使用$?
或直接通過一個if
語句來檢查以保持其簡潔。例如:
if ! mv "${file_list}" "${dest_dir}/" ; then
echo "Unable to move ${file_list} to ${dest_dir}" >&2
exit "${E_BAD_MOVE}"
fi
# Or
mv "${file_list}" "${dest_dir}/"
if [[ "$?" -ne 0 ]]; then
echo "Unable to move ${file_list} to ${dest_dir}" >&2
exit "${E_BAD_MOVE}"
fi
Bash也有 PIPESTATUS
變數,允許檢查從管道所有部分返回的代碼。如果僅僅需要檢查整個管道是成功還是失敗,以下的方法是可以接受的:
tar -cf - ./* | ( cd "${dir}" && tar -xf - )
if [[ "${PIPESTATUS[0]}" -ne 0 || "${PIPESTATUS[1]}" -ne 0 ]]; then
echo "Unable to tar files to ${dir}" >&2
fi
可是,只要你運行任何其他命令, PIPESTATUS
將會被覆蓋。如果你需要基於管道中發生的錯誤執行不同的操作,那麼你需要在運行命令後立即將 PIPESTATUS
賦值給另一個變數(別忘了 [
是一個會將 PIPESTATUS
擦除的命令)。
tar -cf - ./* | ( cd "${DIR}" && tar -xf - )
return_codes=(${PIPESTATUS[*]})
if [[ "${return_codes[0]}" -ne 0 ]]; then
do_something
fi
if [[ "${return_codes[1]}" -ne 0 ]]; then
do_something_else
fi