Shell脚本编写第四季--流程控制

举报
踏雪寻梅2021 发表于 2021/06/20 16:15:52 2021/06/20
【摘要】 一、"if" 表达式。如果条件为真,则执行then后面的部分:if ....; then ....elif ....; then ....else ....fi大多数情况下,可用测试命令来对条件进行测试。比如可以比较字符串、判断文件是否存在及是否可读等等...通常用" [ ] "来表示条件测试。注意要确保方括号的空格。[ -f "somefile" ] :判断是否是一个文件[ -x "/bi...

一、"if" 表达式。如果条件为真,则执行then后面的部分:

if ....; then
 ....
elif ....; then
 ....
else
 ....
fi

大多数情况下,可用测试命令来对条件进行测试。比如可以比较字符串、判断文件是否存在及是否可读等等...通常用" [ ] "来表示条件测试。注意要确保方括号的空格。

[ -f "somefile" ] :判断是否是一个文件
[ -x "/bin/ls" ] :判断/bin/ls是否存在并有可执行权限
[ -n "$var" ] :判断$var变量是否有值
[ "$a" = "$b" ] :判断$a和$b是否相等

执行man test可以查看所有测试表达式可以比较和判断的类型。直接执行以下脚本:

#!/bin/sh
if [ "$SHELL" = "/bin/bash" ]; then
echo "your login shell is the bash (bourne again shell)"
else
echo "your login shell is not bash but $SHELL"
fi

  变量$SHELL包含了登录shell的名称,和/bin/bash进行了比较。C语言使用者会用表达式:[ -f "/etc/shadow" ] && echo "This computer uses shadow passwors"

这里 && 是一个快捷操作符,如果左边的表达式为真则执行右边的语句;也可以认为是逻辑运算中与操作,表示如果 /etc/shadow文件存在则打" This computer uses shadow passwors"。同样这在shell编程中也是可用的,如:

#!/bin/sh
mailfolder=/var/spool/mail/james
[ -r "$mailfolder" ]||{ echo "Can not read $mailfolder"exit 1; }
echo "$mailfolder has mail from:"
grep "^From " $mailfolder

  该脚本首先判断mailfolder是否可读。如果可读则打印该文件中的"From" 行。如果不可读则或操作生效,打印错误信息后脚本退出。这里有个问题,那就是

我们必须有两个命令:-打印错误信息和-退出程序。使用花括号以匿名函数的形式将两个命令放到一起作为一个命令使用。

  不用与和或操作符,也可以用if表达式作任何事情,但是用与或操作符会更便利。

二、case表达式可以用来匹配一个给定的字符串,而不是数字。

case ... in
...) do something here
esac

让我们看一个例子。 file命令可以辨别出一个给定文件的文件类型,比如:file lf.gz

这将返回:lf.gz: gzip compressed data, deflated, original filename, last modified: Mon Aug 27 23:09:18 2001, os: Unix

利用这一点写一个可以自动解压bzip2, gzip 和zip 类型的压缩文件,叫做smartzip的脚本:

#!/bin/sh
ftype=`file "$1"`
case "$ftype" in
"$1: Zip archive"*)
unzip "$1"
"$1: gzip compressed"*)
gunzip "$1"
"$1: bzip2 compressed"*)
bunzip2 "$1"
*) error "File $1 can not be uncompressed with smartzip";;
esac 

在这里使用了一个特殊的变量$1。该变量包含了传递给该程序的第一个参数值。也就是说,当运行:smartzip articles.zip

$1 就是字符串 articles.zip

三、select 表达式是一种bash的扩展应用,尤其擅长于交互式使用。用户可以从一组不同的值中进行选择。

select var in ...do
break
done
.... now $var can be used ....

下面是一个例子:

#!/bin/sh
echo "What is your favourite OS?"
select var in "Linux""Gnu Hurd""Free BSD""Other"; do
break
done
echo "You have selected $var"

下面是该脚本运行的结果:
What is your favourite OS?
1) Linux
2) Gnu Hurd
3) Free BSD
4) Other
#? 1
You have selected Linux

四、您也可以在shell中使用如下的loop表达式:

while ...; do
  ....
done  

while -loop 将运行直到表达式测试为真。 will run while the expression that we test for is true. 关键字"break" 用来跳出循环。而关键字"continue"用来不执行余下的部分而直接跳到下一个循环。

五、for-loop表达式查看一个字符串列表 (字符串用空格分隔) 然后将其赋给一个变量:

for var in ....; do
  ....
done

在下面的例子中,将分别打印ABC到屏幕上:

#!/bin/sh
for var in A B C do
echo "var is $var"
done  

下面是一个更为有用的脚本showrpm,其功能是打印一些RPM包的统计信息:
#!/bin/sh
# list a content summary of a number of RPM packages
# USAGE: showrpm rpmfile1 rpmfile2 ...
# EXAMPLE: showrpm /cdrom/RedHat/RPMS/*.rpm
for rpmpackage in $*; do
if [ -r "$rpmpackage" ];then
echo "=============== $rpmpackage =============="
rpm -qi -p $rpmpackage
else
echo "ERROR: cannot read file $rpmpackage"
fi
done

这里出现了第二个特殊的变量$*,该变量包含了所有输入的命令行参数值。如果您运行 showrpm openssh.rpm w3m.rpm webgrep.rpm

此时 $* 包含了 3 个字符串,即openssh.rpm, w3m.rpm and webgrep.rpm.

六、引号

  在向程序传递任何参数之前,程序会扩展通配符和变量。这里所谓扩展的意思是程序会把通配符(比如*)替换成合适的文件名,它变量替换成变量值。为了防

止程序作这种替换,可用引号:假设在当前目录下有一些文件,两个jpg文件, mail.jpg tux.jpg

#!/bin/sh
echo *.jpg 

这将打印出"mail.jpg tux.jpg"的结果。引号 (单引号和双引号) 将防止这种通配符扩展:

#!/bin/sh
echo "*.jpg"
echo '*.jpg'
这将打印"*.jpg" 两次。单引号更严格一些。它可以防止任何变量扩展。双引号可以防止通配符扩展但允许变量扩展。

#!/bin/sh
echo $SHELL
echo "$SHELL"
echo '$SHELL'
运行结果为:
/bin/bash
/bin/bash
$SHELL


最后,还有一种防止这种扩展的方法,那就是使用转义字符——反斜杆 \:
echo \*.jpg
echo \$SHELL
这将输出:
*.jpg
$SHELL 

七、Here document.

  当要将几行文字传递给一个命令时,here document.一种不错的方法。对每个脚本写一段帮助性的文字是很有用的,此时如果我们四有那个 here document.就不必用echo函数一行行输出。一 "Here document.quot; << 开头,后面接上一个字符串,这个字符串还必须出现在here document.末尾。下面是一个例子,在该例子中,我们对多个文件进行重命名,并且使用here document.打印帮助:

#!/bin/sh
# we have less than 3 arguments. Print the help text:
if [ $# -lt 3 ]then
cat <<HELP
ren -- renames a number of files using sed regular expressions
USAGE: ren 'regexp' 'replacement' files...
EXAMPLE: rename all *.HTM files in *.html:
ren 'HTM$' 'html' *.HTM
HELP
exit 0
fi
OLD="$1"
NEW="$2"
# The shift command removes one argument from the list of
# command line arguments.
shift
shift
# $* contains now all the files:
for file in $*; do
if [ -f "$file" ]then
newfile=`echo "$file" | sed "s/${OLD}/${NEW}/g"`
if [ -f "$newfile" ]; then
echo "ERROR: $newfile exists already"
else
echo "renaming $file to $newfile ..."
mv "$file""$newfile"
fi
fi
done

  这是一个复杂一些的例子。第一个if表达式判断输入命令行参数是否小于3 (特殊变量$# 表示包含参数的个数) 。如果输入参数小于3个,则将帮助文字传递给cat命令,然后由cat命令将其打印在屏幕上。打印帮助文字后程序退出。如果输入参数等于或大于3个,就将第一个参数赋值给变量OLD,第二个参数赋值给变量NEW。下一步,使用shift命令将第一个和第二个参数从参数列表中删除,这样原来的第三个参数就成为参数列表$*的第一个参数。然后开始循环,命令行参数列表被一个接一个地被赋值给变量$file。接着判断该文件是否存在,如果存在则通过sed命令搜索和替换来产生新的文件名。然后将反短斜线内命令结果赋值给newfile。这样得到了旧文件名和新文件名。然后使用mv命令进行重命名。

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。