Bash 备忘录

2021-08-27 17:52 更新

本章为大家总结了 linux bash shell 脚本的快速参考备忘录。

开始

Hello.sh

  1. #!/bin/bash
  2. VAR="world"
  3. echo "Hello $VAR!" # => Hello world!
  4. ## 执行脚本
  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. '

参数
请参阅:特殊参数

  1. $1 $9 # 参数 1 ... 9
  2. $0 # 脚本本身的名称
  3. $1 # 第一个参数
  4. ${10} # 位置参数 10
  5. $# # 参数数量
  6. $$ # shell的进程ID
  7. $* # 所有参数
  8. $@ # 所有参数,从第一个开始
  9. $- # 当前选项
  10. $_ # 上一个命令的最后一个参数

函数
请参阅:函数

  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

外壳执行
请参阅:命令替换

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

Bash 参数扩展

句法

  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

切片
请参阅:参数扩展

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

定义数组

  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[@]} # 总人数
  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

数组作为参数

  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 条件

整数条件

  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 字节
  6. [[ -r FILE ]] # 可读的
  7. [[ -w FILE ]] # 可写的
  8. [[ -x FILE ]] # 可执行的
  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 循环

基本的 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
  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 函数

定义函数

  1. myfunc() {
  2. echo "hello $1"
  3. }
  4. ## 和上面的函数一样 (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)"

引发错误

  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. ## 用于在出现错误时退出
  5. ## 避免级联错误
  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. ## 不区分大小写的全局
  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 #不要立即执行扩展结果

扩展

  1. !$ # 展开最近命令的最后一个参数
  2. !* # 展开最近命令的所有参数
  3. !-n # 展开n最近的命令
  4. !n # 展开n历史命令
  5. !<command> # 展开最近的命令调用 <command>

操作

  1. !! 再次执行上一条命令
  2. !!:s/<FROM>/<TO>/ # 替换最近命令中第一次出现的<FROM>to<TO>
  3. !!:gs/<FROM>/<TO>/ # 替换最近命令中所有出现的<FROM>to<TO>
  4. !$:t # 仅从最近命令的最后一个参数扩展 basename
  5. !$:h # 仅从最近命令的最后一个参数展开目录
  6. ## !!并且!$可以替换为任何有效的扩展。

切片

  1. !!:n # 仅从n最近的命令展开第 th 个标记(命令是0;第一个参数是1)
  2. !^ # 从最近的命令展开第一个参数
  3. !$ # 从最近的命令展开最后一个标记
  4. !!:n-m # 从最近的命令扩展令牌范围
  5. !!:n-$ # 将nth 标记扩展到最近命令的最后一个
  6. ## !!可以使用任何有效的扩展,即取代!cat,!-2,!42等。

其他

数值计算

  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. $? # 上一个任务的退出状态
  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

条件执行

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

严格模式

  1. set -euo pipefail
  2. IFS=$'\n\t'
  3. 请参阅:非官方的 bash 严格模式
以上内容是否对您有帮助:
在线笔记
App下载
App下载

扫描二维码

下载编程狮App

公众号
微信公众号

编程狮公众号