Bash 备忘清单

这是开始使用 linux bash shell 脚本的快速参考备忘单。

入门

hello.sh

  1. #!/bin/bash
  2. VAR="world"
  3. echo "Hello $VAR!" # => Hello world!

执行脚本

  1. $ 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 (关于空间)

注释

  1. # 这是一个内联 Bash 注释。
  2. : '
  3. 这是一个
  4. 非常整洁的评论
  5. 在 bash
  6. '

多行注释使用 :' 打开和 ' 关闭

参数

表示 描述
$1$9 参数 1 … 9
$0 脚本本身的名称
$1 第一个论点
${10} 位置参数 10
$# 参数数量
$$ shell 的进程 id
$* 所有论据
$@ 所有参数,从第一个开始
$- 当前选项
$_ 上一个命令的最后一个参数

见:特殊参数

函数

  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

表示 描述
{A,B} A B 相同
{A,B}.js A.js B.js 相同
{1..5} 1 2 3 4 5 相同

见:大括号扩展

Shell 执行

  1. # => I'm in /当前/的/路径
  2. echo "I'm in $(PWD)"
  3. # Same as:
  4. echo "I'm in `pwd`"

见:命令替换

Bash 参数扩展

语法

代码 描述
${FOO%suffix} 删除后缀
${FOO#prefix} 删除前缀
${FOO%%suffix} 去掉长后缀
${FOO##prefix} 删除长前缀
${FOO/from/to} 替换第一个匹配项
${FOO//from/to} 全部替换
${FOO/%from/to} 替换后缀
${FOO/#from/to} 替换前缀

子字符串

表示 描述
${FOO:0:3} 子串 (位置,长度)
${FOO:(-3):3} 从右边开始的子串

Length

表示 描述
${#FOO} $FOO 的长度

默认值

表示 描述
${FOO:-val} $FOO,如果未设置,则为 val
${FOO:=val} 如果未设置,则将 $FOO 设置为 val
${FOO:+val} val 如果设置了$FOO
${FOO:?message} 如果 $FOO 未设置,则显示消息并退出

替代 Substitution

  1. echo ${food:-Cake} #=> $food or "Cake"
  1. STR="/path/to/foo.cpp"
  2. echo ${STR%.cpp} # /path/to/foo
  3. echo ${STR%.cpp}.o # /path/to/foo.o
  4. echo ${STR%/*} # /path/to
  5. echo ${STR##*.} # cpp (extension)
  6. echo ${STR##*/} # foo.cpp (basepath)
  7. echo ${STR#*/} # path/to/foo.cpp
  8. echo ${STR##*/} # foo.cpp
  9. echo ${STR/foo/bar} # /path/to/bar.cpp

切片 Slicing

  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"
  1. BASEPATH=${SRC##*/}
  2. echo $BASEPATH # => "foo.cpp"
  3. DIRPATH=${SRC%$BASEPATH}
  4. echo $DIRPATH # => "/path/to/"

Transform

  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 数组

定义数组

  1. Fruits=('Apple' 'Banana' 'Orange')
  2. Fruits[0]="Apple"
  3. Fruits[1]="Banana"
  4. Fruits[2]="Orange"
  5. ARRAY1=(foo{1..2}) # => foo1 foo2
  6. ARRAY2=({A..D}) # => A B C D
  7. # 合并 => foo1 foo2 A B C D
  8. ARRAY3=(${ARRAY1[@]} ${ARRAY2[@]})
  9. # 声明构造
  10. declare -a Numbers=(1 2 3)
  11. Numbers+=(4 5) # 附加 => 1 2 3 4 5

索引

:- -
${Fruits[0]} 第一个元素
${Fruits[-1]} 最后一个元素
${Fruits[*]} 所有元素
${Fruits[@]} 所有元素
${#Fruits[@]} 总数
${#Fruits} 第一节长度
${#Fruits[3]} 第n个长度
${Fruits[@]:3:2} 范围
${!Fruits[@]} 所有 Key

迭代 Iteration

  1. Fruits=('Apple' 'Banana' 'Orange')
  2. for e in "${Fruits[@]}"; do
  3. echo $e
  4. done

With index

  1. for i in "${!Fruits[@]}"; do
  2. printf "%s\t%s\n" "$i" "${Fruits[$i]}"
  3. done

操作

  1. Fruits=("${Fruits[@]}" "Watermelon") # 推
  2. Fruits+=('Watermelon') # 也推
  3. Fruits=( ${Fruits[@]/Ap*/} ) # 通过正则表达式匹配删除
  4. unset Fruits[2] # 删除一项
  5. Fruits=("${Fruits[@]}") # 复制
  6. Fruits=("${Fruits[@]}" "${Veggies[@]}") # 连接
  7. lines=(`cat "logfile"`) # 从文件中读取

数组作为参数

  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
  1. sounds[dog]="bark"
  2. sounds[cow]="moo"
  3. sounds[bird]="tweet"
  4. 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

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

Bash 条件句

整数条件

条件 描述
[[ NUM -eq NUM ]] 等于 Equal
[[ NUM -ne NUM ]] 不等于 Not equal
[[ NUM -lt NUM ]] 少于 Less than
[[ NUM -le NUM ]] 小于或等于 Less than or equal
[[ NUM -gt NUM ]] 大于 Greater than
[[ NUM -ge NUM ]] 大于或等于 Greater than or equal
(( NUM < NUM )) 少于
(( NUM <= NUM )) 小于或等于
(( NUM > NUM )) 比…更棒
(( NUM >= NUM )) 大于或等于

字符串条件

条件 描述
[[ -z STR ]] 空字符串
[[ -n STR ]] 空字符串
[[ STR == STR ]] 平等的
[[ STR = STR ]] 相等(同上)
[[ STR < STR ]] 小于 (ASCII)
[[ STR > STR ]] 大于 (ASCII)
[[ STR != STR ]] 不相等
[[ STR =~ STR ]] 正则表达式

例子

字符串

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

组合

  1. if [[ X && Y ]]; then
  2. ...
  3. fi

相等

  1. if [[ "$A" == "$B" ]]; then
  2. ...
  3. fi

正则表达式

  1. if [[ '1. abc' =~ ([a-z]+) ]]; then
  2. echo ${BASH_REMATCH[1]}
  3. fi

更小

  1. if (( $a < $b )); then
  2. echo "$a is smaller than $b"
  3. fi

存在

  1. if [[ -e "file.txt" ]]; then
  2. echo "file exists"
  3. fi

文件条件

条件 描述
[[ -e FILE ]] 存在
[[ -d FILE ]] 目录
[[ -f FILE ]] 文件
[[ -h FILE ]] 符号链接
[[ -s FILE ]] 大小 > 0 字节
[[ -r FILE ]] 可读
[[ -w FILE ]] 可写
[[ -x FILE ]] 可执行文件
[[ f1 -nt f2 ]] f1 比 f2 新
[[ f1 -ot f2 ]] f2 早于 f1
[[ f1 -ef f2 ]] 相同的文件

更多条件

条件 描述
[[ -o noclobber ]] 如果启用 OPTION
[[ ! EXPR ]] 不是 Not
[[ X && Y ]] 和 And
`[[ 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

Bash 循环

基本 for 循环

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

类似 C 的 for 循环

  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

具有步长

  1. for i in {5..50..5}; do
  2. echo "Welcome $i"
  3. 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

```bash {data=3,5} for number in $(seq 1 3); do if [[ $number == 2 ]]; then continue; fi echo “$number” done

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

Until

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

永远

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

永远(简写)

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

正在读取行

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

Bash 函数

定义函数

  1. myfunc() {
  2. echo "hello $1"
  3. }
  1. # 同上(替代语法)
  2. function myfunc() {
  3. echo "hello $1"
  4. }
  1. myfunc "John"

返回值

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

正在引发错误

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

Bash 选项

选项

  1. # 避免覆盖文件
  2. # (echo "hi" > foo)
  3. set -o noclobber
  4. # 用于出错时退出
  5. # 避免级联错误
  6. set -o errexit
  7. # 揭示隐藏的失败
  8. set -o pipefail
  9. # 公开未设置的变量
  10. set -o nounset

全局选项

  1. # 不匹配的 glob 被删除
  2. # ('*.foo' => '')
  3. shopt -s nullglob
  4. # 不匹配的 glob 抛出错误
  5. shopt -s failglob
  6. # 不区分大小写的球体
  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 历史

命令

命令 描述
history 显示历史
sudo !! 使用 sudo 运行上一个命令
shopt -s histverify 不要立即执行扩展结果

表达式

表达式 描述
!$ 展开最新命令的最后一个参数
!* 展开最新命令的所有参数
!-n 展开第 n 个最近的命令
!n 展开历史中的第 n 个命令
!<command> 展开最近调用的命令 <command>

操作

代码 描述
!! 再次执行最后一条命令
!!:s/<FROM>/<TO>/ 在最近的命令中将第一次出现的 <FROM> 替换为 <TO>
!!:gs/<FROM>/<TO>/ 在最近的命令中将所有出现的 <FROM> 替换为 <TO>
!$:t 仅从最近命令的最后一个参数扩展基本名称
!$:h 仅从最近命令的最后一个参数展开目录

!!!$ 可以替换为任何有效的扩展。

切片 Slices

代码 描述
!!:n 仅扩展最近命令中的第 n 个标记(命令为 0;第一个参数为 1
!^ 从最近的命令展开第一个参数
!$ 从最近的命令中展开最后一个标记
!!:n-m 从最近的命令扩展令牌范围
!!:n-$ 从最近的命令中将第 n 个标记展开到最后

!! 可以替换为任何有效的扩展,即 !cat!-2!42 等。

各种各样的

数值计算

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

子 shell

  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 # 标准输出到(文件)
  2. python hello.py >> output.txt # 标准输出到(文件),追加
  3. python hello.py 2> error.log # 标准错误到(文件)
  4. python hello.py 2>&1 # 标准错误到标准输出
  5. python hello.py 2>/dev/null # 标准错误到(空null)
  6. python hello.py &>/dev/null # 标准输出和标准错误到(空null)
  1. python hello.py < foo.txt # 将 foo.txt 提供给 python 的标准输入

来源相对

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

脚本目录

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

Case/switch

  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

或者

  1. traperr() {
  2. echo "ERROR: ${BASH_SOURCE[1]} at about ${BASH_LINENO[0]}"
  3. }
  4. set -o errtrace
  5. trap traperr ERR

printf

  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 "看来您的互联网连接正常"
  3. fi

特殊变量

表达式 描述
$? 最后一个任务的退出状态
$! 最后一个后台任务的 PID
$$ shell PID
$0 shell 脚本的文件名

特殊参数

grep 检查

  1. if grep -q 'foo' ~/.bash_history; then
  2. echo "您过去似乎输入过“foo”"
  3. fi

反斜杠转义

  •  
  • !
  • \”
  • #
  • \&
  • \’
  • (
  • )
  • \,
  • \;
  • \<
  • >
  • [
  • |
  • \
  • ]
  • \^
  • {
  • }
  • `
  • \$
  • *
  • \?

使用 \ 转义这些特殊字符

Heredoc

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

转到上一个目录

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

读取输入

  1. echo -n "Proceed? [y/n]: "
  2. read ans
  3. echo $ans
  1. read -n 1 ans # 只有一个字符

条件执行

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

严格模式

  1. set -euo pipefail
  2. IFS=$'\n\t'

参见:非官方 bash 严格模式

可选参数

  1. args=("$@")
  2. args+=(foo)
  3. args+=(bar)
  4. echo "${args[@]}"

将参数放入数组中,然后追加

另见