Bash 備忘錄

2021-08-27 17:52 更新

本章為大家總結了 linux bash shell 腳本的快速參考備忘錄。

開始

Hello.sh

  1. #!/bin/bash
  2. VAR="world"
  3. echo "Hello $VAR!" # => Hello world!
  4. ## 執(zhí)行腳本
  5. $ bash hello.sh

變量

  1. NAME="John"
  2. echo ${NAME} # => John
  3. echo $NAME # => John
  4. echo "$NAME" # => John
  5. echo '$NAME' # => $NAME
  6. echo "${NAME}!" # => John!
  7. NAME = "John" # => Error (about space)

注釋

  1. ## 單行注釋
  2. : '
  3. This is a
  4. very neat comment
  5. in bash
  6. '
  7. :'
  8. 多行注釋以 [:'] 開始, ['] 結束
  9. '

參數(shù)
請參閱:特殊參數(shù)

  1. $1 $9 # 參數(shù) 1 ... 9
  2. $0 # 腳本本身的名稱
  3. $1 # 第一個參數(shù)
  4. ${10} # 位置參數(shù) 10
  5. $# # 參數(shù)數(shù)量
  6. $$ # shell的進程ID
  7. $* # 所有參數(shù)
  8. $@ # 所有參數(shù),從第一個開始
  9. $- # 當前選項
  10. $_ # 上一個命令的最后一個參數(shù)

函數(shù)
請參閱:函數(shù)

  1. get_name() {
  2. echo "John"
  3. }
  4. echo "You are $(get_name)"

條件句
請參閱:條件

  1. if [[ -z "$string" ]]; then
  2. echo "String is empty"
  3. elif [[ -n "$string" ]]; then
  4. echo "String is not empty"
  5. fi

花括號擴展
請參閱:花括號擴展

  1. echo {A,B}.js
  2. {A,B} # 與...一樣 A B
  3. {A,B}.js # 與...一樣 A.js B.js
  4. {1..5} # 與...一樣 1 2 3 4 5

外殼執(zhí)行
請參閱:命令替換

  1. echo "I'm in $(PWD)"
  2. ## 上下的表示是一樣的
  3. echo "I'm in `pwd`"

Bash 參數(shù)擴展

句法

  1. ${FOO%suffix} # 刪除后綴
  2. ${FOO#prefix} # 刪除前綴
  3. ${FOO%%suffix} # 去除長后綴
  4. ${FOO##prefix} # 刪除長前綴
  5. ${FOO/from/to} # 替換第一場比賽
  6. ${FOO//from/to} # 全部替換
  7. ${FOO/%from/to} # 替換后綴
  8. ${FOO/#from/to} # 替換前綴
  1. ## 子串
  2. ${FOO:0:3} # 子串(位置,長度)
  3. ${FOO:(-3):3} # 右邊的子串
  1. ## 長度
  2. ${#FOO} # $FOO的長度
  1. ## 替換
  2. echo ${food:-Cake} #=> $food or "Cake"
  3. STR="/path/to/foo.cpp"
  4. echo ${STR%.cpp} # /path/to/foo
  5. echo ${STR%.cpp}.o # /path/to/foo.o
  6. echo ${STR%/*} # /path/to
  7. echo ${STR##*.} # cpp (extension)
  8. echo ${STR##*/} # foo.cpp (basepath)
  9. echo ${STR#*/} # path/to/foo.cpp
  10. echo ${STR##*/} # foo.cpp
  11. echo ${STR/foo/bar} # /path/to/bar.cpp

切片
請參閱:參數(shù)擴展

  1. name="John"
  2. echo ${name} # => John
  3. echo ${name:0:2} # => Jo
  4. echo ${name::2} # => Jo
  5. echo ${name::-1} # => Joh
  6. echo ${name:(-1)} # => n
  7. echo ${name:(-2)} # => hn
  8. echo ${name:(-2):2} # => hn
  9. length=2
  10. echo ${name:0:length} # => Jo

基本路徑和目錄路徑

  1. SRC="/path/to/foo.cpp"
  2. BASEPATH=${SRC##*/}
  3. echo $BASEPATH # => "foo.cpp"
  4. DIRPATH=${SRC%$BASEPATH}
  5. echo $DIRPATH # => "/path/to/"

轉換

  1. STR="HELLO WORLD!"
  2. echo ${STR,} # => hELLO WORLD!
  3. echo ${STR,,} # => hello world!
  4. STR="hello world!"
  5. echo ${STR^} # => Hello world!
  6. echo ${STR^^} # => HELLO WORLD!
  7. ARR=(hello World)
  8. echo "${ARR[@],}" # => hello world
  9. echo "${ARR[@]^}" # => Hello World

Bash 數(shù)組

定義數(shù)組

  1. Fruits=('Apple' 'Banana' 'Orange')
  2. Fruits[0]="Apple"
  3. Fruits[1]="Banana"
  4. Fruits[2]="Orange"
  5. ARRAY2=(foo{1..2}) # => foo1 foo2
  6. ARRAY3=({A..D}) # => A B C D
  7. ## declare construct
  8. declare -a Numbers=(1 2 3 4 5 6)

索引

  1. ${Fruits[0]} # 第一個元素
  2. ${Fruits[-1]} # 最后一個元素
  3. ${Fruits[*]} # 所有元素
  4. ${Fruits[@]} # 所有元素
  5. ${#Fruits[@]} # 總人數(shù)
  6. ${#Fruits} # 長度 1
  7. ${#Fruits[3]} # 第 n 個長度
  8. ${Fruits[@]:3:2} # 范圍
  9. ${!Fruits[@]} # 所有的鑰匙

迭代

  1. Fruits=('Apple' 'Banana' 'Orange')
  2. for e in "${Fruits[@]}"; do
  3. echo $e
  4. done
  1. ## 帶索引
  2. for i in "${!Fruits[@]}"; do
  3. printf "%s\t%s\n" "$i" "${Fruits[$i]}"
  4. done

操作

  1. Fruits=("${Fruits[@]}" "Watermelon") # Push
  2. Fruits+=('Watermelon') # Also Push
  3. Fruits=( ${Fruits[@]/Ap*/} ) # Remove by regex match
  4. unset Fruits[2] # Remove one item
  5. Fruits=("${Fruits[@]}") # Duplicate
  6. Fruits=("${Fruits[@]}" "${Veggies[@]}") # Concatenate
  7. lines=(`cat "logfile"`) # Read from file

數(shù)組作為參數(shù)

  1. function extract()
  2. {
  3. local -n myarray=$1
  4. local idx=$2
  5. echo "${myarray[$idx]}"
  6. }
  7. Fruits=('Apple' 'Banana' 'Orange')
  8. extract Fruits 2 # => Orangle

Bash 字典

定義

  1. declare -A sounds
  2. sounds[dog]="bark"
  3. sounds[cow]="moo"
  4. sounds[bird]="tweet"
  5. sounds[wolf]="howl"

使用字典

  1. echo ${sounds[dog]} # Dog's sound
  2. echo ${sounds[@]} # All values
  3. echo ${!sounds[@]} # All keys
  4. echo ${#sounds[@]} # Number of elements
  5. unset sounds[dog] # Delete dog

迭代

  1. for val in "${sounds[@]}"; do
  2. echo $val
  3. done
  4. for key in "${!sounds[@]}"; do
  5. echo $key
  6. done

Bash 條件

整數(shù)條件

  1. [[ NUM -eq NUM ]] # 相等
  2. [[ NUM -ne NUM ]] # 不相等
  3. [[ NUM -lt NUM ]] # 小于
  4. [[ NUM -le NUM ]] # 小于或等于
  5. [[ NUM -gt NUM ]] # 大于
  6. [[ NUM -ge NUM ]] # 大于或等于
  7. (( NUM < NUM )) # 小于
  8. (( NUM <= NUM )) # 小于或等于
  9. (( NUM > NUM )) # 大于
  10. (( NUM >= NUM )) # 大于或等于

字符串條件

  1. [[ -z STR ]] # 空字符串
  2. [[ -n STR ]] # 非空字符串
  3. [[ STR == STR ]] # 相等
  4. [[ STR = STR ]] # 相等(同上)
  5. [[ STR < STR ]] # 小于(ASCII)
  6. [[ STR > STR ]] # 大于(ASCII)
  7. [[ STR != STR ]] # 不相等
  8. [[ STR =~ STR ]] # 正則表達式

文件條件

  1. [[ -e FILE ]] # 存在
  2. [[ -d FILE ]] # 目錄
  3. [[ -f FILE ]] # 文件
  4. [[ -h FILE ]] # 符號鏈接
  5. [[ -s FILE ]] # 文件大小 > 0 字節(jié)
  6. [[ -r FILE ]] # 可讀的
  7. [[ -w FILE ]] # 可寫的
  8. [[ -x FILE ]] # 可執(zhí)行的
  9. [[ f1 -nt f2 ]] # f1 比 f2 更新
  10. [[ f1 -ot f2 ]] # f2 比 f1 更舊
  11. [[ f1 -ef f2 ]] # 相同的文件

更多條件

  1. [[ -o noclobber ]] # 如果啟用了 OPTION
  2. [[ ! EXPR ]] # Not
  3. [[ X && Y ]] # And
  4. [[ X || Y ]] # Or

邏輯和,或

  1. if [ "$1" = 'y' -a $2 -gt 0 ]; then
  2. echo "yes"
  3. fi
  4. if [ "$1" = 'n' -o $2 -lt 0 ]; then
  5. echo "no"
  6. fi

例子

  1. ## 細繩
  2. if [[ -z "$string" ]]; then
  3. echo "String is empty"
  4. elif [[ -n "$string" ]]; then
  5. echo "String is not empty"
  6. else
  7. echo "This never happens"
  8. fi
  9. ## 組合
  10. if [[ X && Y ]]; then
  11. ...
  12. fi
  13. ## 平等的
  14. if [[ "$A" == "$B" ]]; then
  15. ...
  16. fi
  17. ## 正則表達式
  18. if [[ '1. abc' =~ ([a-z]+) ]]; then
  19. echo ${BASH_REMATCH[1]}
  20. fi
  21. ## 較小
  22. if (( $a < $b )); then
  23. echo "$a is smaller than $b"
  24. fi
  25. ## 存在
  26. if [[ -e "file.txt" ]]; then
  27. echo "file exists"
  28. fi

Bash 循環(huán)

基本的 for 循環(huán)

  1. for i in /etc/rc.*; do
  2. echo $i
  3. done

類似 C 的 for 循環(huán)

  1. for ((i = 0 ; i < 100 ; i++)); do
  2. echo $i
  3. done

范圍

  1. for i in {1..5}; do
  2. echo "Welcome $i"
  3. done
  4. ## 有步長
  5. for i in {5..50..5}; do
  6. echo "Welcome $i"
  7. done

自動遞增

  1. i=1
  2. while [[ $i -lt 4 ]]; do
  3. echo "Number: $i"
  4. ((i++))
  5. done

自動遞減

  1. i=3
  2. while [[ $i -gt 0 ]]; do
  3. echo "Number: $i"
  4. ((i--))
  5. done

Continue

  1. for number in $(seq 1 3); do
  2. if [[ $number == 2 ]]; then
  3. continue;
  4. fi
  5. echo "$number"
  6. done

Break

  1. for number in $(seq 1 3); do
  2. if [[ $number == 2 ]]; then
  3. # Skip entire rest of loop.
  4. break;
  5. fi
  6. # This will only print 1
  7. echo "$number"
  8. done

Util

  1. count=0
  2. until [ $count -gt 10 ]; do
  3. echo "$count"
  4. ((count++))
  5. done

Forever

  1. while true; do
  2. # here is some code.
  3. done

Forever(簡寫)

  1. while :; do
  2. # here is some code.
  3. done

閱讀線

  1. cat file.txt | while read line; do
  2. echo $line
  3. done

Bash 函數(shù)

定義函數(shù)

  1. myfunc() {
  2. echo "hello $1"
  3. }
  4. ## 和上面的函數(shù)一樣 (alternate syntax)
  5. function myfunc() {
  6. echo "hello $1"
  7. }
  8. myfunc "John"

返回值

  1. myfunc() {
  2. local myresult='some value'
  3. echo $myresult
  4. }
  5. result="$(myfunc)"

引發(fā)錯誤

  1. myfunc() {
  2. return 1
  3. }
  4. if myfunc; then
  5. echo "success"
  6. else
  7. echo "failure"
  8. fi

Bash 選項

選項

  1. ## 免覆蓋文件
  2. ## (echo "hi" > foo)
  3. set -o noclobber
  4. ## 用于在出現(xiàn)錯誤時退出
  5. ## 避免級聯(lián)錯誤
  6. set -o errexit
  7. ## 顯示隱藏的失敗
  8. set -o pipefail
  9. ## 公開未設置的變量
  10. set -o nounset

全局選項

  1. ## 將刪除不匹配的全局
  2. ## ('*.foo' => '')
  3. shopt -s nullglob
  4. ## 不匹配的全局拋出錯誤
  5. shopt -s failglob
  6. ## 不區(qū)分大小寫的全局
  7. shopt -s nocaseglob
  8. ## 通配符匹配點文件
  9. ## ("*.sh" => ".foo.sh")
  10. shopt -s dotglob
  11. ## 允許**進行遞歸匹配
  12. ## ('lib/**/*.rb' => 'lib/a/b/c.rb')
  13. shopt -s globstar

Bash 歷史

命令

  1. history # 顯示歷史
  2. shopt -s histverify #不要立即執(zhí)行擴展結果

擴展

  1. !$ # 展開最近命令的最后一個參數(shù)
  2. !* # 展開最近命令的所有參數(shù)
  3. !-n # 展開n最近的命令
  4. !n # 展開n歷史命令
  5. !<command> # 展開最近的命令調用 <command>

操作

  1. !! 再次執(zhí)行上一條命令
  2. !!:s/<FROM>/<TO>/ # 替換最近命令中第一次出現(xiàn)的<FROM>to<TO>
  3. !!:gs/<FROM>/<TO>/ # 替換最近命令中所有出現(xiàn)的<FROM>to<TO>
  4. !$:t # 僅從最近命令的最后一個參數(shù)擴展 basename
  5. !$:h # 僅從最近命令的最后一個參數(shù)展開目錄
  6. ## !!并且!$可以替換為任何有效的擴展。

切片

  1. !!:n # 僅從n最近的命令展開第 th 個標記(命令是0;第一個參數(shù)是1)
  2. !^ # 從最近的命令展開第一個參數(shù)
  3. !$ # 從最近的命令展開最后一個標記
  4. !!:n-m # 從最近的命令擴展令牌范圍
  5. !!:n-$ # 將nth 標記擴展到最近命令的最后一個
  6. ## !!可以使用任何有效的擴展,即取代!cat,!-2,!42等。

其他

數(shù)值計算

  1. $((a + 200)) # Add 200 to $a
  2. $(($RANDOM%200)) # Random number 0..199

Subshell

  1. (cd somedir; echo "I'm now in $PWD")
  2. pwd # still in first directory

檢查命令

  1. command -V cd
  2. #=> "cd is a function/alias/whatever"

重定向

  1. python hello.py > output.txt # stdout to (file)
  2. python hello.py >> output.txt # stdout to (file), append
  3. python hello.py 2> error.log # stderr to (file)
  4. python hello.py 2>&1 # stderr to stdout
  5. python hello.py 2>/dev/null # stderr to (null)
  6. python hello.py &>/dev/null # stdout and stderr to (null)
  7. python hello.py < foo.txt # feed foo.txt to stdin for python

對來源相

  1. source "${0%/*}/../share/foo.sh"

腳本目錄

  1. DIR="${0%/*}"

案例/開關

  1. case "$1" in
  2. start | up)
  3. vagrant up
  4. ;;
  5. *)
  6. echo "Usage: $0 {start|stop|ssh}"
  7. ;;
  8. esac

陷阱錯誤

  1. trap 'echo Error at about $LINENO' ERR
  2. ## 或者
  3. traperr() {
  4. echo "ERROR: ${BASH_SOURCE[1]} at about ${BASH_LINENO[0]}"
  5. }
  6. set -o errtrace
  7. trap traperr ERR

打印輸出

  1. printf "Hello %s, I'm %s" Sven Olga
  2. #=> "Hello Sven, I'm Olga
  3. printf "1 + 1 = %d" 2
  4. #=> "1 + 1 = 2"
  5. printf "Print a float: %f" 2
  6. #=> "Print a float: 2.000000"

獲取選項

  1. while [[ "$1" =~ ^- && ! "$1" == "--" ]]; do case $1 in
  2. -V | --version )
  3. echo $version
  4. exit
  5. ;;
  6. -s | --string )
  7. shift; string=$1
  8. ;;
  9. -f | --flag )
  10. flag=1
  11. ;;
  12. esac; shift; done
  13. if [[ "$1" == '--' ]]; then shift; fi

檢查命令的結果

  1. if ping -c 1 google.com; then
  2. echo "It appears you have a working internet connection"
  3. fi

格雷普檢查

  1. if grep -q 'foo' ~/.bash_history; then
  2. echo "You appear to have typed 'foo' in the past"
  3. fi

特殊變量

  1. $? # 上一個任務的退出狀態(tài)
  2. $! # 最后一個后臺任務的PID
  3. $$ # shell 的PID
  4. $0 # shell 腳本的文件名

反斜杠轉義

  1. # &
  2. ' ( ) ,
  3. ; < > [
  4. | \ ] ^
  5. { } ` $
  6. * ?
  7. ## 轉義這些特殊字符 \

Heredoc

  1. cat <<END
  2. hello world
  3. END

讀取輸入

  1. echo -n "Proceed? [y/n]: "
  2. read ans
  3. echo $ans
  4. read -n 1 ans # Just one character

轉到上一個目錄

  1. pwd # /home/user/foo
  2. cd bar/
  3. pwd # /home/user/foo/bar
  4. cd -
  5. pwd # /home/user/foo

條件執(zhí)行

  1. git commit && git push
  2. git commit || echo "Commit failed"

嚴格模式

  1. set -euo pipefail
  2. IFS=$'\n\t'
  3. 請參閱:非官方的 bash 嚴格模式
以上內(nèi)容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號