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来设置,否则可能会将结果截断为整数