书写优雅的shell脚本(一)- if语句

时间:2021-02-19 15:33:59

使用unix/linux的程序人员几乎都写过shell脚本,但这其中很多人都是为了完成功能而在网上找代码段,这样写出来的shell脚本在功能方面当然是没有什么问题,但是这样的方式不能写出优雅的shell脚本。

从今天开始,starfe就将自己平时在书写shell脚本过程中的经历做一总结,力图形成一个系列---“书写优雅的shell脚本”。

在此,对“优雅”一词的定义有4点:(1)健壮;(2)结构清晰;(3)性能好;(4)力求简单。


好了,废话不多说,开始今天的主题:if语句

1. if判断式

(1)

if [ 条件判断一 ] &&(||) [ 条件判断二 ]; then

xxx

fi

(2)

if [ 条件判断一 ] &&(||) [ 条件判断二 ]; then

xxx

else

xxx

fi

(3)

if [ 条件判断一 ] &&(||) [ 条件判断二 ]; then

xxx

elif [ 条件判断三 ] &&(||) [ 条件判断四 ]; then

xxx

...

else 

xxxx

fi

2. if中的二元比较

2.1 整数比较

     -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 ]

     注:以上其实不是健壮的代码,上面这些代码在有些情况下会存现错误提示,而真正健壮的是使用双括号来表示,即if [[ $a -eq $b ]]

这是为何?做个测试如下:

$>a=;b=3;

$>if [ a -gt $b ]; then echo "true";fi

-bash: [: -gt: unary operator expected  //报错

  而改为$>if [[ a > 3 ]]; then echo "true";fi  //不再报错,这也是我们所期望的

究其原因,是因为如果变量a值为空(由于shell是弱类型语言,对变量赋值都是当字符串对待),那么就成了 [ -gt 3 ] ,显然 [ 和 $b 不相等并且缺少了 [ 符号,所以报了这样的错误。当然不总是出错,如果变量a值不为空,程序就正常了,所以这样的错误还是很隐蔽的。

    或者用下面的方法也能避免这种错 误:if [ "$a"x -gt "$b"x ]; then   echo "true"; fi。当然,x也可以是其他字符。顺便提一点,shell中有没有双引号在很多情况下是一致的。

2.2 字符串比较

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

不等于,如:if [ $a != $b ]

大于,在ASCII字母顺序下.如:if [ $a \> $b ]

小于,在ASCII字母顺序下.如:if [ $a \< $b ] 注意:要使用转义符“\”。

2.3 文件比较

[ 文件1 -nt 文件2 ] 为真 如果 文件1 has been changed more recently than 文件2, or 如果文件1 存在 而且文件2 does not。
[ 文件1 -ot 文件2 ] 为真 如果 文件1 比 文件2 旧, 或者 文件2 存在而且 文件1 不存在。
[ 文件1 -ef 文件2 ] 为真 如果 文件1 而且 文件2 refer to the same device 而且 inode numbers。

2.4 表达式比较

[ 表达式1 -a 表达式2 ] 如果表达式1 而且表达式2 同时为真则为真 。
[ 表达式1 -o 表达式2 ] 如果表达式1 或者表达式2 其中之一为真则为真。

3.if中的一元比较

[ -a 文件 ] 如果文件存在为真。
[ -b 文件 ] 如果 文件 存在 而且 是一个 块-特殊 文件为真。
[ -c 文件 ] 为真 如果 文件 存在 而且 是一个 字-特殊 文件。
[ -d 文件 ] 为真 如果 文件 存在 而且 是一个 目录。
[ -e 文件 ] 为真 如果 文件 存在。
[ -f 文件 ] 为真 如果 文件 存在 而且 是一个 普通 文件。
[ -g 文件 ] 为真 如果 文件 存在 而且 已经设置了他的 SGID 位。
[ -h 文件 ] 为真 如果 文件 存在 而且 是一个 符号连接。
[ -k 文件 ] 为真 如果 文件 存在 而且 他的粘住位已经设置。
[ -p 文件 ] 为真 如果 文件 存在 而且 是一个 已经命名的管道 (F 如果O)。
[ -r 文件 ] 为真 如果 文件 存在 而且 是可读的。
[ -s 文件 ] 为真 如果 文件 存在 而且 比零字节大。
[ -t FD ] 为真 如果 文件 文件描述符已经打开 而且 指向一个终端。
[ -u 文件 ] 为真 如果 文件 存在 而且 已经设置了他的 SUID (set user ID)位。
[ -w 文件 ] 为真 如果 文件 为真 如果 文件 存在 而且 是可写的。
[ -x 文件 ] 为真 如果 文件 存在 而且 是可执行的。
[ -O 文件 ] 为真 如果 文件 存在 而且 属于有效用户ID。
[ -G 文件 ] 为真 如果 文件 存在 而且 属于有效组ID。
[ -L 文件 ] 为真 如果 文件 存在 而且 是一个 符号连接。
[ -N 文件 ] 为真 如果 文件 存在 而且 has been mod 如果ied since it was last read。
[ -S 文件 ] 为真 如果 文件 存在 而且 是一个 socket。

[ -o 选项名 ] 为真 如果 shell 选项 "选项名" 开启。

[ -z STRING ] 为真 如果 "STRING"的长度是零。
[ -n STRING ] 或者 [ STRING ] 为真 "STRING"的长度是非零值。

[ ! EXPR ] 如果EXPR 为假则为真。
[ ( EXPR ) ] 返回EXPR 的值。 这样可以用来忽略正常的操作符优先级。