基本操作符
分号
可以用于分割命令或语句
竖线
管道,将左边命令的标准输出作为右侧命令的标准输入,如:
echo "replace_me will change the world" | sed "s:replace_me:I:g"
会输出 I will change the world
右箭头
重定向标准输出,单箭头>为替换,双箭头>>为添加,如:
echo "new text" > new.txt
echo "append text" >> append.txt
echo "print error" 2>&1
echo "depress error" 2>/dev/null
左箭头
双箭头<<,类似管道,只是将<<后面内容作为标准输入传递给指定程序
其用法为:
×command× << [\]×TEXT_BLOCK_TAG×
×TEXT×
×TEXT_BLOCK_TAG×
其作用为将×TEXT×
的内容作为标准输入传递给×command×
,其中方括号[]中的反斜线作用为禁止展开×TEXT×
中的变量或命令
例如:
cat << FLAG
$(echo null-str: \"${NULLSTR}\" is NULL), root dir is: `realpath /`
FLAG
将输出"null-str: "" is NULL, root dir is: /"
而
cat << \FLAG
$(echo null-str: \"${NULLSTR}\" is NULL), root dir is: `realpath /`
FLAG
则会输出"$(echo null-str: \"${NULLSTR}\" is NULL), root dir is: `realpath /`"
test命令
test 是一个 shell 内建命令,基本用法为test ×表达式×
,其返回值为 0 或 1,存储在$?中。0 表示真, 1 表示假。
在×表达式×
中,变量${VAR}尽量加双引号"",这是为了避免${VAR}中间有空格或其它特殊字符会影响表达式
×表达式×可以是任意字符串或空字符串,也可以是使用test支持的选项连接的字符串
表达式
test ×表达式×
中,test将对<<表达式>>进行如下测试:
表达式 # 表达式为真
! # 表达式为假
-a # 逻辑与 表达式1 –a 表达式2 两个表达式都为真
-o # 逻辑或 表达式1 –o 表达式2 两个表达式有一个为真
检查字符串
test 检查字符串时,非空的字符串返回真,空字符串或者没有参数都返回假(字符串所构成的命令不能成功执行)。
if test "${UNDEFINED}"; then
echo "UNDEFINED没有被定义,这句不会被输出"
else
echo "UNDEFINED没有被定义,其构成空字符串,test结果为假,该句被输出"
fi
test支持下列字符串检查选项:
-n # 字符串的长度为非零
-z # 字符串的长度为零
= # 字符串相等,不是“`==`”,因为“`=`”是`posix`标准**
!= # 字符串不等
例如:
if test -n "${UNDEFINED}"; then
echo "UNDEFINED构成的字符串长度为0,这句不会被输出"
else
echo "UNDEFINED构成的字符串长度为0,test结果为假,所以这句会被输出"
fi
if test -z "${UNDEFINED}"; then
echo "UNDEFINED构成空字符串,test结果为真,这句将被输出"
fi
SAME="same"
if test "same" = "${SAME}"; then
echo "变量SAME构成的字符串与same相同,test结果为真,该句将被输出"
fi
if test "different" != "${SAME}"; then
echo "变量SAME构成的字符串与different不同,test结果为真,该句将被输出."
fi
比较数字(整数)
-eq # 整数1 -eq 整数2 如果 整数1 等于 整数2,则为真
-ge # 整数1 -ge 整数2 如果 整数1 大于或等于 整数2,则为真
-gt # 整数1 -gt 整数2 如果 整数1 大于 整数2,则为真
-le # 整数1 -le 整数2 如果 整数1 小于或等于 整数2,则为真
-lt # 整数1 -lt 整数2 如果 整数1 小于 整数2,则为真
-ne # 整数1 -ne 整数2 如果 整数1 不等于 整数2,则为真
检查文件属性
-ef # File1 –ef File2 两个文件具有同样的设备号和i结点号
-nt # File1 –nt File2 文件1比文件2新
-ot # File1 –ot File2 文件1比文件2旧
-b #文件存在并且是块设备文件。
-c #文件存在并且是字符设备文件。
-d #文件存在并且是一个目录。
-e #文件存在。
-f #文件存在并且是一般文件。
-g #文件存在并且设置了组ID。
-G #文件存在并且属于有效组ID
-h # 文件存在并且是一个符号链接(同-L)
-L # 文件存在并且是一个符号链接(同-h)
-k # 文件存在并且设置了sticky位
-o # 文件存在并且属于有效用户ID
-p # 文件存在并且是一个命名管道
-r # 文件存在并且可读
-s # 文件存在并且有内容
-t # FD 文件描述符是在一个终端打开的
-u # 文件存在并且设置了它的用户ID位(set-user-id)
-w # 文件存在并且可写
-x # 文件存在并且可执行
-S # 文件存在并且是一个 socket。
参考
內建命令
[...] (中括号,方括号)
[[...]] (双中括号,双方括号)
((...)) (双括号)
(...) (单括号)
单中括号 [ ×表达式× ]
左右单中括号[]与×表达式×之间有空格
单中括号与test命令等价
if [ -e /usr ];then echo "/usr will always exist"; fi
与
if test -e /usr;then echo "/usr will always exist"; fi
效果相同
双中括号[[ ×表达式× ]]
左右双中括号[[]]与×表达式×之间有空格
双中括号支持符号转义,可以再其中使用诸如<,>,==,!=,&&,||等符号,注意表达式中字符与符号之间要有空格
例如:
[[ 1 < 0 || "1" == "1" && 1 > 0 && "12" < "2" && 1 != 0 && "11" != "1" && "11" > "1" ]] ; echo $?
这行代码将输出0,也就是测试结果为真
在双中括号中,>与-gt效果一致,<与-lt效果一致,==与=效果一致,!=与-ne效果一致,&&与-a效果一致,||与-o效果一致
双圆括号(( ×表达式× ))
用于数学计算,只支持整数,不支持浮点数。左右双圆括号(())与中间的×表达式×之间有空格
非数字字符将被当做变量处理,默认值为0
0与逻辑假都使得$?为1即假,非0与逻辑真都使得$?为0即真
例如:
(( 2 * 3 == 6 && 1 + 2 == 3 )); echo $? # 结果为0,即真。
(( 2 ? 0 : 1 )); echo $? #支持三目运算,结果为1,即假。0为假,非0为真
(( undefined_var )); echo $? #undefined_var被当成未初始化变量,其值为0,所以结果为假。
defined_var = 5; ((defined_var++));echo ${defined_var} # 对变量defined_var进行数学计算
# 循环,输出0,1,2,3,4
for (( i = 0; i < 5; i++)); do
echo $i;
done
单圆括号()
- 启动一个有独立的环境变量的 subshell 来执行括号内的代码
A=2;(export A=1; echo ${A}); echo ${A} # 将分两行输出1 2,subshell的环境变量不影响外部shell的环境变量
- 在()前加$可以捕获subshell的输出转换成字符串
A=$(echo hello);echo ${A} # 将输出hello,echo hello在subshell中被执行,其输出被捕获为字符串赋给变量A
- 初始化数组
ARRAY=(a b c d)
echo ${ARRAY[0]} # 输出a
echo ${ARRAY[3]} # 输出d
echo ${ARRAY[4]} # 无输出
for i in ${ARRAY[@]}; # 也可以用${ARRAY[*]}
do
echo ${i} # 将分4行输出a b c d
done
# ${#ARRAY[@]}表示数组长度
for (( i = 0 ; i < ${#ARRAY[@]} ; i++ ))
do
# 将分4行输出
# ARRAY[0] = a
# ARRAY[1] = b
# ARRAY[2] = c
# ARRAY[3] = d
echo ARRAY[${i}] = ${ARRAY[${i}]}
done
参考
if 关键字
标准的if语句的语法不含中括号
if ×命令×; then
# command in yes
else
# command in no
fi
其中×命令×可以是任何命令或可执行程序,if判断该命令运行后$?中的内容
该命令可以是test或者类test內建命令,如[],[[]],(())
if ls; then
echo ls without arguments will always succeed
fi
if ls /nodir 2>/dev/null; then
echo this branch will not be reached
else
echo call ls on an invalid path will always fail
fi