程序开发语言,分为两种,一种是解释型语言,一种是编译型语言。解释型语言是解释语句,并且能根据流程控制机制让语句按需执行,解释一条语句就返回语句的结果,这种语言需要一个解释器,而linux中的bash就是这样的一个解释器,常见的解释型语言有perl,python,ruby,bash。编译型语言,首先需要编译源程序,并且能够将源程序转换为二进制格式,而后让其执行,这种语言需要一个编辑器,如linux下的gcc就是一个编辑器,编译型语言有C,C++,C#等。无论是解释型语言还是编译型语言,都有语言控制结构,没有控制结构的程序,不能算做一个好的程序。
语言控制结构分为三种:
顺序执行:默认法则,逐条执行各语句
选择执行:条件判断,只有部分是符合条件的,只执行符合条件的部分
循环执行:将同一段代码反复的执行n次
这篇博客写关于选择控制结构。
linux的选择控制结构有两种一种是 if-then,一种是case--esac。
1、if-then
bash条件测试:
[ expression ]
[[ expression ]]
test expression
bash命令
if-then有三种结构形式:
单分支if语句
if [ 条件 ];then
语句...
fi
这种单分支的语句,当条件满足的时候,就会运行then后面的语句,不满足就直接退出判断语句
eg:系统中是否存在mysql用户,存在就显示mysql exist.
#!/bin/bash #this shell test mysql user exist system. if id mysql ;then echo "mysql exist." fi
双分支if语句
if [ 条件 ];then
语句...
else
语句...
fi
这种双分支的语句,等条件满足的时候就会运行then后面的语句,条件不满足的时候就运行else后面的语句。
eg1:如果指定的用户存,先说明其已经存在,并显示其ID号和SHELL;否则,就添加用户,并显示其ID号;
#!/bin/bash Username=mysql if id $Username &>/dev/null;then Id=`grep "^$Username\>" /etc/passwd | cut -d: -f3` Shell=`grep "^$Username\>" /etc/passwd | cut -d: -f7` echo "$Username exist,ID is $Id,shell is $Shell." else useradd $Username Id=`grep "^$Username\>" /etc/passwd | cut -d: -f3` echo "$Username ID is $Id." fi
eg2:判断用户是否存在,存在则显示ID和SHELL,不存在就添加用户,并显示其ID。
分析:这题与上一个列子不一样的地方是这个用户没有指定,既然没有指定,就需要自己手动去shell一个参数,而脚本怎么去引用参数。这里就用到了bash变量中的位置变量
位置变量:$1,$2.....$9($1就是脚本传递的第一个参数...)
$0:脚本自身名称
[email protected]:所有位置参数的列表
$*:所有位置参数
$#:位置参数的个数
#!/bin/bash if id $1 &>/dev/null;then Id=`grep "^$1\>" /etc/passwd | cut -d: -f3` Shell=`grep "^$1\>" /etc/passwd | cut -d: -f7` echo "$1 exist,ID is $Id,shell is $Shell." else useradd $1 Id=`grep "^$1\>" /etc/passwd | cut -d: -f3` echo "$1 ID is $Id." fi
运行脚本 bash eg2.sh mysql (eg.sh是脚本的名称,mysql是传递给脚本的第一个参数)
eg3:通过参数传递一系列用户名给脚本,让脚本添加这些用户;但要先判断用户是否存在,不存在而后再添加;添加完成后,显示一共添加了几个用户;当然,不能包括因为事先存在而没有添加的;
分析:这个脚本不明确的是不知道要传递几个参数,所以用位置变量有点不符合要求,所以引用了[email protected]这个特殊的变量,然后这些用户,你得一个一个的去取这些用户,然后进行判断是否存在,用循环控制了如何一个一个的取这些用户,然后用户不能重复的一个一个的取,所以用户得取一个T掉[email protected]列表中的用户,用了shift命令,shift命令就是轮流的T除用户。
#!/bin/bash count=0 for user in [email protected];do if id $1 &>/dev/null;then echo "user $1 exsit." shift else useradd $1 count=$[$count+1] echo "add $1." shift fi done echo "total users:$count"
运行脚本
嵌套if语句(最常见的嵌套)
if [ 条件1 ];then
if [ 条件2 ];then
语句...
else
语句.....
fi
else
语句....
fi
这种嵌套的if语句就相对复杂一点,当条件1满足时,就运行第一个then后面的语句,在条件1满足后,在判断条件2,当条件1和条件2同时满足的时候就会运行第二个then后面的语句,当只有条件1满足,条件2不满足就会运行第一个else后面的语句,如果条件1都不满足,那就直接运行第二个else后面的语句了。
eg4:判断一个用户,如果存在则判断是否是普通用户,还是系统用户,还是admin用户。不存在则显示该用户不存在。
分析:首先判断用户是否存在,存在则比较用户的ID是否为大于等于500,大于等于500则表示为普通用户,是否1-499,是则表示为系统用户,是否为0,是则为admin用户。
本例子还用到了数值的比较,然后还用到了条件的组合
linux test命令支持数值比较、字符串比较、文件比较 详情请man test
数值比较
num1 -eq num2 检查num1是否等于num2
num1 -ge num2 检查num1是否>或等于num2
num1 -gt num2 检查num1是否大于num2
num1 -le num2 检查num1是否<等于num2
num1 -lt num2 检查num1是否<num2
num1 -ne num2 检查num1是否不等于num2
组合条件测试
-a:与 [ $Uid -ge 1 -a $Uid -le 499 ]
-o:或 [ $Uid -eq 0 -a $Uid -ge 500 ]
!:非,单目操作符 [ ! $Uid -eq 0 ]
#!/bin/bash if id $1 &> /dev/null;then Id=`grep "^$1\>" /etc/passwd | cut -d: -f3` if [ $Id -ge 500 ];then echo "$1 is common user." elif [ $Id -lt 500 -a $Id -ge 1 ];then echo "$1 is system user." else echo "$1 is admin user." fi else echo "$1 Not exist." fi
脚本执行结果
eg5:判定用户的shell是否为登录shell;
分析:首先判断用户的shell是否存在,redhat中有一个用户的shell就为空,如果shell存在则判断shell是否为bash,如果是则表示为登录用户,如果不是则表示为不能登录用户。在redhat 5.x的版本上。
字符串比较
str1 = str2 检查str1与str2是否相同
str1 != str2 检查str1与str2是否不同
str1 < str2 检查str1是否小于str2
str1 > str2 检查str1是否大于str2
-n str1 检查str1的长度是否大于0
-z str1 检查str1的长度是否为0
=~:判断左边的字符串是否能够被右边的模式所匹配,通常用于[[]]; [[ "$opt1" =~ pattern ]],
一般锚定行首和行尾
#!/bin/bash Shell=`grep "^$1:" /etc/passwd | cut -d: -f7` if [ -z $Shell ];then echo "$1 user no shell" else if [ "$Shell" == "/bin/bash" ]; then echo "$1 user shell is login shell." else echo "$1 user shell is nologin shell." fi fi
运行结果
eg6:判断当前主机的CPU生产商,其信息在/proc/cpuinfo文件中vendor id一行中。如果其生产商为GenuineIntel,就显示其为Intel公司;否则,就显示其为AMD公司;
#!/bin/bash Vendor=`grep "vendor_id" /proc/cpuinfo | uniq | cut -d: -f2` if [[ "$Vendor" =~ [[:space:]]*GenuineIntel$ ]]; then echo "Intel" else echo "AMD" fi
运行结果
[[email protected] test]# bash eg6.sh
Intel
eg7:
写一个脚本:可以接受一个参数,其使用形式如下:
script.sh {start|stop|restart|status}
如果参数为start,创建空文件/var/lock/subsys/script,并显示“Starting script successfully.”;
如果参数为stop,则删除文件/var/lock/subsys/script,并显示“Stop script finished.”;
如果参数为restart,则删除文件/var/lock/subsys/script后重新创建,并显示“Restarting script successfully.”;
如果参数为status,那么:
如果/var/lock/subsys/script文件存在,则显示为“script is running.”
否则,则显示为“script is stopped.”
其它任何参数:则显示“script.sh {start|stop|restart|status}”
文件比较
-b file 检查文件是否存在且是一个块特殊文件
-c file 检查文件是否存在且是一个字符文件
-d file 检查file是否存在并且是一个目录
-e file 检查文件是否存在
-f file 检查文件是否存在并且是一个文件
-h file 检查文件存在且为一个符合链接
-r file 检查文件是否存在并且可读
-s file 检查文件是否存在且不为空
-w file 检查文件是否存在且可写
-x file 检查文件是否存在且可执行
-O file 检查文件是否存在且别当前用户拥有
-G file 检查是否存在并且默认组是否为当前用户组
#!/bin/bash #Author:litaotao dir=`basename $0` if [ $1 == "start" ];then touch /var/lock/subsys/$dir echo "Starting script successfully." elif [ $1 == "stop" ];then rm /var/lock/subsys/$dir &> /dev/null echo "Stop script finished." elif [ $1 == "restart" ];then rm /var/lock/subsys/$dir &> /dev/null touch /var/lock/subsys/$dir echo "Restarting script sucessfully." elif [ $1 == "status" ];then if [ -e /var/lock/subsys/$dir ];then echo "script is running." else echo "script is stopped." fi else echo "$dir.sh {statr|stop|restart|status}" fi
运行方式 bash eg6.sh start 各位博友可以自己把shell copy后执行一遍分析结果
======博主所学知识来着于恩师马哥的亲授======
马哥教育“2014夏令营”开始啦!!!马哥教育是目前性价比最高的Linux培训,国内好评度排名第一,并被网友称为Linux界的"黄埔军校",全部课程采用Centos6.5x86_64讲解,经过几期网络班的总结和锤炼,逐渐完善的课程体系,学员学习进度监督和优质的考试系统检验学员掌握程度,活跃的在线答疑环节,名师陪伴,牛人指点,精彩不容错过。
详情猛戳:http://www.magedu.com/
课程内容:http://www.magedu.com/mentuqc
网络班一体化服务:http://mageedu.blog.51cto.com/4265610/1379598
联系方式:客服QQ
2813150558 客服QQ 1661815153
============================================
shell脚本编程之选择控制结构