前言
只要讲到程序的话,那么条件判断式,即是
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
语句,返回零值的退出状态。因此elif
的then
代码块中的语句得以执行。删除了/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
部分会被执行。
参考资料
- 结构化命令之 if-then;if-then-else语句
- 《鸟哥的Linux私房菜》
- 《Linux命令行与shell脚本编程大全》