Shell語法規範

不羈的羅恩發表於2020-06-14

背景

部落格: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放在同一行,donewhile/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個空格縮排可選項。
  • 在同一行可選項的模式右圓括號之後和結束符 ;;之前各需要一個空格。
  • 長可選項或者多命令可選項應該被拆分成多行,模式、操作和結束符;;在不同的行。

匹配表示式比caseesac 縮排一級。多行操作要再縮排一級。一般情況下,不需要引用匹配表示式。模式表示式前面不應該出現左括號。避免使用;&;;&符號。即:

# 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

相關文章