bash技巧

基本操作符

分号

可以用于分割命令或语句

竖线

管道,将左边命令的标准输出作为右侧命令的标准输入,如:

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。

参考

说说 bash 的 if 语句
Bash : test 命令

內建命令

[...]   (中括号,方括号)
[[...]] (双中括号,双方括号)
((...)) (双括号)
(...)   (单括号)

单中括号 [ ×表达式× ]

左右单中括号[]与×表达式×之间有空格
单中括号与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

单圆括号()

  1. 启动一个有独立的环境变量的 subshell 来执行括号内的代码
A=2;(export A=1; echo ${A}); echo ${A} # 将分两行输出1 2,subshell的环境变量不影响外部shell的环境变量
  1. 在()前加$可以捕获subshell的输出转换成字符串
A=$(echo hello);echo ${A}  # 将输出hello,echo hello在subshell中被执行,其输出被捕获为字符串赋给变量A
  1. 初始化数组
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

参考

说说 bash 的 if 语句
Shell 数组
Bash脚本遍历数组

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

参考:

说说 bash 的 if 语句

你可能感兴趣的:(bash技巧)