if和比较

if和比较

多数语言,一般比较运算常用在判断,控制流程中。所以呢,本文记录一下shell中的逻辑运算跟if控制流程。

if结构

提示,如果then要是另起一行,则不用加分号。这其实是shell的语法,多行命令,每个都是独立的一个命令,则不需要行尾添加

# 形式一
if [] ;then
fi
# 形式二
if [];then
else
fi

# 形式3
if [];then
elif [];then
.....
else
fi
# 其他 比如嵌套

shell参数

  • $0 Shell本身的文件名
  • $1 Shell的第一个位置参数,一直到$9;当n>=10时,需要使用${n}来获取参数
  • $# 传递到脚本的参数个数
  • $* 以一个单字符串显示所有向脚本传递的参数
  • $$ 脚本运行的当前进程ID号
  • $! 后台运行的最后一个进程的ID号
  • $@ 与$*相同,但是使用时加引号,并在引号中返回每个参数。
  • $- 显示Shell使用的当前选项,与set命令功能相同。
  • $? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。

举例

#!/bin/bash
msg=''
# 判断脚本参数个数,注意,不传参是0,而不是1。
if [ $# -gt 0 ]
  then
  msg="$1"
  # 上一步执行结果可能是数字,也可能是字符,所以呢,下面判断,直接用 = 判断,而不能用-eq,会报错。
  if [ $msg = 0 ]
    then
    msg="ok"
  fi
fi
# 注意下面,双引号里面的内容才会被解析,所以呢,json用了双引号
curl -H 'Content-Type: application/json' \
    -d "{\"msgtype\": \"text\", \"text\": {\"content\": \"主人,任务结束。!!!\n执行: $msg\"}}" \
    https://oapi.dingtalk.com/robot/send?access_token=

比较运算

整数比较

-eq       等于,如:if [ "$a" -eq "$b" ]
-ne       不等于,如:if [ "$a" -ne "$b" ]
-gt       大于,如:if [ "$a" -gt "$b" ]
-ge       大于等于,如:if [ "$a" -ge "$b" ]
-lt       小于,如:if [ "$a" -lt "$b" ]
-le       小于等于,如:if [ "$a" -le "$b" ]
<         小于(需要双括号),如:(("$a" < "$b"))
<=        小于等于(需要双括号),如:(("$a" <= "$b"))
>         大于(需要双括号),如:(("$a" > "$b"))
>=        大于等于(需要双括号),如:(("$a" >= "$b"))

说明:

整数运算不能跟字符运算混合在一起,如判断相等,如下

#!/bin/bash

a="hahaha"
b=0
if [ $a -eq $b ]
then
   echo "true"
else
   echo "false"
fi

提示错误:

./demo.sh: line 5: [: hahaha: integer expression expected

如果确实需要将,字符相等,直接使用=号。

字符串比较

=       等于,如:if [ "$a" = "$b" ]
   ==     等于,如:if [ "$a" == "$b" ], 与=等价

   !=      不等于,如:if [ "$a" != "$b" ], 这个操作符将在[[]]结构中使用模式匹配.
   <       小于,在ASCII字母顺序下.如:
              if [[ "$a" < "$b" ]]
              if [ "$a" \< "$b" ]     在[]结构中"<"需要被转义.
   >       大于,在ASCII字母顺序下.如:
          if [[ "$a" > "$b" ]]
          if [ "$a" \> "$b" ]  在[]结构中">"需要被转义.
   -z       空串。字符串为"null".就是长度为0.
   -n       字符串不为"null"

注意:==的功能在[[]]和[]中的行为是不同的,如下:


[[ $a == z* ]]    # 如果$a以"z"开头(模式匹配)那么将为true
[[ $a == "z*" ]] # 如果$a等于z*(字符匹配),那么结果为true

[ $a == z* ]      # File globbing 和word splitting将会发生
[ "$a" == "z*" ] # 如果$a等于z*(字符匹配),那么结果为true

另外,字符比较常见:

https://www.jb51.net/article/56559.htm

if参数

  • –b 当file存在并且是块文件时返回真
  • -c 当file存在并且是字符文件时返回真
  • -d 当pathname存在并且是一个目录时返回真
  • -e 当pathname指定的文件或目录存在时返回真
  • -f 当file存在并且是正规文件时返回真
  • -g 当由pathname指定的文件或目录存在并且设置了SGID位时返回为真
  • -h 当file存在并且是符号链接文件时返回真,该选项在一些老系统上无效
  • -k 当由pathname指定的文件或目录存在并且设置了“粘滞”位时返回真
  • -p 当file存在并且是命令管道时返回为真
  • -r 当由pathname指定的文件或目录存在并且可读时返回为真
  • -s 当file存在文件大小大于0时返回真
  • -u 当由pathname指定的文件或目录存在并且设置了SUID位时返回真
  • -w 当由pathname指定的文件或目录存在并且可执行时返回真。一个目录为了它的内容被访问必然是可执行的。
  • -o 当由pathname指定的文件或目录存在并且被子当前进程的有效用户ID所指定的用户拥有时返回真。
  • -x 文件可执行。
  • L file为符号链接。
  • -nt 前者比后者文件新。则为真
  • -ol 前者比后者文件旧。则为真。

调用形式:

#!/bin/bash
# 一个参数
#这里的-x 参数判断$myPath是否存在并且是否具有可执行权限
if [ ! -x "$mypath" ]; then
    mkdir "$mypath"
fi

# 两个参数
if [ $file1 -nt $file2 ]; then
    echo "new"
fi

比较举例

方法一: if [ ${A} -lt ${B} ]; then …
这是最基本的比较方法,使用lt(小于),gt(大于),le(小于等于),ge(大于等于),优点:还没发现;缺点:只能比较整数,使用lt,gt等不直观

方法二: if ((${A} < ${B})) then …
这是CShell风格比较,优点:不用使用lt,gt等难记的字符串;缺点:还是只能比较整数

方法三: if (echo ${A} ${B} | awk ‘!($1>$2){exit 1}’) then …
这是使用awk比较,优点:可以比较小数;缺点:表达式太复杂,难记

方法四: if (echo ${A} - ${B} | bc -q | grep -q “^-“); then …
这是使用bc计算比较,优点:可以比较小数;缺点:表达式更复杂,难记

计算举例

​ 方法一:typeset C=$(expr ${A} + ${B});
SHELL中的基本工具,优点:方便检测变量是否为数字;缺点:只能计算整数,且只能计算加减法,不能计算乘除法

方法二:let “C=${A}+${B}”; 或 let “C=A+B”
内嵌命令计算,优点:能计算乘除法及位运算等;缺点:只能计算整数

方法三:typeset C=$((A+B))
CShell风格的计算,优点:能计算乘除法及位运算等,简介,编写方便;缺点:不能计算小数

方法四:typeset C=${echo ${A} ${B} | awk ‘{print $1+$2}’)
使用awk计算,优点:能计算小数,可以实现多种计算方式,计算灵活;缺点:表达式太复杂

方法五:typeset C=${echo ${A} + ${B} | bc -q)
使用awk计算,优点:能计算小数,计算方式比awk还多,计算灵活;缺点:表达式太复杂,小数点后面的位数必须使用scale=N来设置,否则可能会将结果截断为整数