shell脚本编程之条件判断式(一)

前言

只要讲到程序的话,那么条件判断式,即是if-then这种判别式肯定一定要学习。因为很多时候,我们都必须依据某些数据来判断程序该如何进行。

if-then语句

最基本的条件判断式是if-then语句。 if-then语句有如下格式:

if command
then
commands
fi

如果该命令的退出状态码是0(该命令成功运行),位于then部分的命令就会被执行。如果该命令的退出状态码是其他值, then部分的命令就不会被执行, shell会继续执行脚本中的下一个命令。 fi语句用来表示if-then语句到此结束。

#!/bin/zsh
#testing the if-then statement
if pwd
then
        echo It worked
fi

这个脚本在if行采用了pwd命令(显然会成功)。如果命令成功结束,echo语句就会显示该文本字符串。在命令行运行该脚本时,会得到如下结果:

$ ./test1.sh
/mnt/f/linux/zsh-script/if-then
It worked

下面是另一个例子

#!/bin/zsh
#testing the if-then statement
if iloveu
then
        echo It worked
fi
echo We are outside the statement

这次我们故意执行了一个不存在的命令iloveu,所以它会产生一个非零的退出状态码,shell会跳过then部分的echo语句。还要注意,运行if语句中的那个错误命令所生成的错误消息依然会显示在脚本的输出中。

if中的命令没有被成功执行时,shell会自动跳过then部分的命令,直接执行该部分后的内容,在该例中即为语句echo We are outside the statement

另一种形式

if-then语句的另一种形式:

if command; then
commands
fi

通过把分号放在待求值的命令尾部,就可以将then语句放在同一行上了。在then部分,你可以使用不止一条命令,可以像在脚本中的其他地方一样在这里列出多条命令。

if-then的两种形式本质上没有区别,只是表现形式有所不同,后者在多条语句时会更加美观。

#!/bin/zsh
#testing multiple commands in the then section
testuser=lyr
if grep $testuser /etc/passwd
then
        echo "This is my first command"
        echo "This is my second command"
        echo "I can even put in other commands besides echo: "
        ls -a /home/$testuser/.b*
fi

if语句行使用grep命令在/etc/passwd文件中查找某个用户名当前是否在系统上使用。如果有用户使用了那个登录名,脚本会显示一些文本信息并列出该用户HOME目录的bash文件。

注意到上述脚本中我们先定义了一个用户变量testuser,之后使用命令替换的方式引用了这个变量,此时的$testuser应该看为lyr。

运行结果:

$ ./test3.sh
lyr:x:1000:1000:,,,:/home/lyr:/usr/bin/zsh
This is my first command
This is my second command
I can even put in other commands besides echo:
/home/lyr/.bash_history  /home/lyr/.bash_logout  /home/lyr/.bashrc

但是,如果将testuser变量设置成一个系统上不存在的用户,则什么都不会显示。

#!/bin/zsh
#testing multiple commands in the then section
testuser=hqw    #hqw用户名并不存在
if grep $testuser /etc/passwd
then
        echo "This is my first command"
        echo "This is my second command"
        echo "I can even put in other commands besides echo: "
        ls -a /home/$testuser/.b*
fi
$ ./test3.sh

如果在这里显示的一些消息可说明这个用户名在系统中未找到,这样可能就会显得更友好,可以用if-then语句的另外一个特性来做到这一点。

if-then-else语句

if-then语句中,不管命令是否成功执行,你都只有一种选择。如果命令返回一个非零退出状态码,shell会跳过该部分继续执行脚本中的下一条命令。在这种情况下,如果能够执行另一组命令就好了。这正是if-then-else语句的作用。

if command
then
commands
else
commands
fi

if语句中的命令返回退出状态码0时, then部分中的命令会被执行,当if语句中的命令返回非零退出状态码时, shell会执行else部分中的命令。现在可以复制并修改测试脚本来加入else部分。

#!/bin/zsh
#testing the else function                                                                                              testuser=NoSuchUser
if grep $testuser /etc/passwd
then
        echo "The bash files for user $testuser are: "
        ls -a /home/$testuser/.b*
        echo
else
        echo "The user $testuser does not exist on this system."
        echo
fi
$ ./test4.sh
The user NoSuchUser does not exist on this system.

这样就更友好了。跟then部分一样, else部分可以包含多条命令。

嵌套if

要检查/etc/passwd文件中是否存在某个用户名以及该用户的目录是否尚在,可以使用嵌套的if-then语句。嵌套的if-then语句位于主if-then-else语句的else代码块中。

#!/bin/zsh
#testing nested ifs                                                                                                     testuser=NoSuchUser
if grep $testuser /etc/passwd
then
        echo "The bash files for user $testuser are: "                                                                  
else
        echo "The user $testuser does not exist on this system."                                                                
       if                   
              ls -d /home/$testuser                                                                                           
       then                                                                                                                            
              echo "However,$testuser has a directory"                                                                        
       fi                                                                                                              
fi
$ ./test5.sh
The user NoSuchUser does not exist on this system.
/home/NoSuchUser/
However, NoSuchUser has a directory.

这个脚本准确无误地发现,尽管登录名已经从/etc/passwd中删除了,但是该用户的目录仍然存在。在脚本中使用这种嵌套if-then语句的问题在于代码不易阅读,很难理清逻辑流程。可以使用else部分的另一种形式: elif。这样就不用再书写多个if-then语句了。elif使用另一个if-then语句延续else部分。

if command1
then
commands
elif command2
then
more commands
fi

elif语句行提供了另一个要测试的命令,这类似于原始的if语句行。如果elif后命令的退出状态码是0,则bash会执行第二个then语句部分的命令。

$ cat test5.sh
#!/bin/zsh
# Testing nested ifs - use elif
testuser=NoSuchUser
if grep $testuser /etc/passwd
then
echo "The user $testuser exists on this system."
elif ls -d /home/$testuser
then
echo "The user $testuser does not exist on this system."
echo "However, $testuser has a directory."
fi
$ ./test5.sh
/home/NoSuchUser
The user NoSuchUser does not exist on this system.
However, NoSuchUser has a directory.

甚至可以更进一步,让脚本检查拥有目录的不存在用户以及没有拥有目录的不存在用户。这可以通过在嵌套elif中加入一个else语句来实现。

$ cat test5.sh
#!/bin/zsh
# Testing nested ifs - use elif & else
testuser=NoSuchUser
if grep $testuser /etc/passwd
then
echo "The user $testuser exists on this system."
elif ls -d /home/$testuser
then
echo "The user $testuser does not exist on this system."
echo "However, $testuser has a directory."
else
echo "The user $testuser does not exist on this system."
echo "And, $testuser does not have a directory."
fi
$ ./test5.sh
/home/NoSuchUser
The user NoSuchUser does not exist on this system.
However, NoSuchUser has a directory.
$ sudo rmdir /home/NoSuchUser
$ ./test5.sh
ls: cannot access /home/NoSuchUser: No such file or directory
The user NoSuchUser does not exist on this system.
And, NoSuchUser does not have a directory.

在/home/NoSuchUser目录被删除之前,这个测试脚本执行的是elif语句,返回零值的退出状态。因此elifthen代码块中的语句得以执行。删除了/home/NoSuchUser目录之后, elif语句返回的是非零值的退出状态。这使得elif块中的else代码块得以执行。记住,在elif语句中,紧跟其后的else语句属于elif代码块。它们并不属于之前的if-then代码块。
可以继续将多个elif语句串起来,形成一个大的if-then-elif嵌套组合。

if command1
then
command set 1
elif command2
then
command set 2
elif command3
then
command set 3
elif command4
then
command set 4
fi

每块命令都会根据命令是否会返回退出状态码0来执行。记住, shell会依次执行if语句,只有第一个返回退出状态码0的语句中的then部分会被执行。

参考资料

  1. 结构化命令之 if-then;if-then-else语句
  2. 《鸟哥的Linux私房菜》
  3. 《Linux命令行与shell脚本编程大全》

你可能感兴趣的:(shell脚本编程之条件判断式(一))