λ:
限定范围
#!/bin/bash
set -euf # 开启选项
# set -x -o pipefail
# set +euf # 关闭选项
...
大部分教程到这就开始写脚本逻辑部分了,但是第二行set -euf可能比第一行还有用,而且第一行也不是非写不可,bash <script_name>.sh 也可以跑。
#!声明脚本文件的解释器- 常用解释器是
sh(bash, zsh), 除此之外之外还有csh(tcsh),ash,ksh等等,是有语法差异的,而且就算是sh, bash, zsh也不是完全的子集和包含关系。所以一定要确定好解释器和环境 cat /etc/shells查看支持的解释器- 现在好多linux发行版
/bin/sh只是/bin/bash软链接,执行时相当于bash --posix。(MacOS还是实际程序,并非软链,不知道是不是UNIX,BSD都这样)
- 常用解释器是
# 注释内容注释set,-开启选项,+关闭选项-e (-o errexit)若指令传回值不等于0,则立即退出shell-u (-o nounset)当执行时使用到未定义过的变量,则显示错误信息。-f (-o noglob)Disable pathname expansion. 路径名禁止使用通配符-x (-o xtrace)打印正在执行的语句-o pipefail管道连接的命令(cmd1|cmd2|cmd3),会有subshell问题,后边命令不受-e影响, 也就是说其中有命令失败时,结果仍会传递给下一条命令。
bash -euf <script_name>.sh也可以开启选项echo $-查看当前开启的选项
set是bash的内建命令,可以help set查看具体用法man builtin了解内建命令- 如果使用未定义的变量,值会是
""空串,这在路径拼接时尤为危险:rm -rf $WORKSPACE/, 如果WORKSPACE是个未定义变量,这条语句直接变成rm -rf /
命令 & 组合命令
man bash
# 打开手册后搜索 SHELL GRAMMAR,
/SHELL GRAMMAR
简单命令 Simple Commands
A simple command is a sequence of optional variable assignments followed by blank-separated words and redirections, and terminated by a control operator.
简单命令是一个由若干
可选变量组成的序列,参数之间空格或者重定向符连接,并且由控制操作符终止控制操作符:
|| & && ; ;; ( ) | <newline>
echo "hello world" > file1.txt- 第一个变量
echo为要执行的命令,同时会被作为参数0传递。 "hello world", file1.txt为命令参数,>重定向符- 返回值是它的退出状态,0为成功(true)。如果命令被信号 n 终止,则返回 128+n
管道 Pipelines
A pipeline is a sequence of one or more commands separated by the character .
[time [-p]] [ ! ] command [ | command2 ... ]
| 管道是一个或多个命令的组成的序列,由字符 | 分隔。 方括号中为可选参数,所以 一条命令也算作是个管道 |
- 前一个命令的标准输出会做为下一个命令的标准输入
- 会产生
subshell,每条指令都会在新的子shell中执行。 - 返回值为最后一条命令执行的结果。 如果设置了
pipefail选项,则其中一条命令失败时直接结束并返回结果。
# example
cat /etc/shells | grep /bin
命令列表 Lists
A list is a sequence of one or more pipelines separated by one of the operators ;, &, &&, or , and optionally terminated by one of ;, &, or <newline>.
简单讲就是一串管道(命令)。
&&,||优先级相同。若以此符号结尾,后边必须要有另一条命令。pipe1 || pipe2 && pipe3, 和其他语言类似,会有截断效果:若cmd1成功则之后cmd2 && cmd3不执行,如果cmd2失败,cmd3不执行。
;,&优先级相同。次于&&, ||pipe; pipe2; pipe3;, 以;结尾的命令,命令依次执行,整条List的返回值为最后一条命令的值。cmd1& cmd2& cmd3&, 以&结尾的命令,会产生subshell, 也就是新起一个子shell执行该命令,并且当前shell不会等待子shell执行完毕,直接返回0。
<换行符>,最常见的结尾。
复合命令 Compound Commands
各种括号
(list), 会产生subshell, list在subshell中执行。{ list; },组命令。list在当前shell执行,返回值为list的返回值。- list中的每条命令必须以
;或者换行符结尾。 list;与{ }之间必须空格分割。{ }是两个保留字,所以整条语句在能识别保留字的区域内才有效。
- list中的每条命令必须以
((expression)),expression:算数表达式,参考表达式求值。- 等同于
let expression
- 等同于
[[ expression ]],expression:条件表达式, 参考表达式求值。[[ ]]是两个保留字,所以整条语句在能识别保留字的区域内才有效。expression与[[ ]]要以空格分割。
for
for name [ [ in [ word ... ] ] ; ] do list ; done
变量
变量名=值, 等号两边不加空格,否则变量名会当成指令,=和值是参数。${变量名}或$变量名取值,${}为了截断变量名和上下文,如${str1}2, 没有花括号变量名会识别为str12
str1="string1"
echo "str1: $str1"
echo "str2: ${str1}2"
数组
shell 数组更像 map<int, T>
# 数组, 数组名=("val1" "val2" "val3")
empty_arr=() # 空数组
arr=("val1" "val2" "val3 3 3")
# 使用:
${arr[50]} # 取值,如果不存在就是空字符串
arr[100]="new val" # 赋值
${#arr[@]} # 当前元素个数
引号,转义字符
shell 操作基本都是在做字符串处理。环境变量里IFS(Internal Field Seprator)为当前分隔符的值。打印一下默认值:
# 由于全是转义字符,用 od 命令处理一下,否则看不到
echo $IFS | od -c
# output:
# 0000000 \t \n \0 \n
# 0000005
echo $IFS | od -a
# output:
# 0000000 sp ht nl nul nl
# 0000005
所以如果字符串中空格,tab,换行等,会被识别成两个字符串。