任务列表:
20.1 shell脚本介绍
20.2 shell脚本结构和执行
20.3 date命令用法
20.4 shell脚本中的变量
20.5 shell脚本中的逻辑判断
20.6 文件目录属性判断
20.7 if特殊用法
20.8/20.9 case判断
20.10 for循环
20.11/20.12 while循环
20.13 break跳出循环
20.14 continue结束本次循环
20.15 exit退出整个脚本
20.1 shell脚本介绍
- shell是一种脚本语言(个人认为:就是自动逐行执行执行命令的过程)
- 关注aming_linux微信公众号,里有shell练习
- blog.lishiming.net阿铭的博客里有相关的内容
- 可以使用逻辑判断、循环等语法
- 可以自定义函数(可以减少代码的重复)
- shell是系统命令的集合
- shell脚本可以实现自动化运维,能大大增加我们的运维效率。(定时执行,比如半夜备份数据库)
- 编写shell之前要有一个思路,也就是方向来决定编写的过程。
20.2 shell脚本结构和执行
- shell脚本开头的固定格式
#!/bin/bash
- 正文件第二行开始。以#开头,则这一整行内容不会被shell执行。一般用于解释说明。
- 脚本的文件名以.sh结尾,用于区分这是一个shell脚本
脚本的执行方法
- 第一种:
chmod +x 1.sh; ./1.sh
- 为文件增加执行权限。然后输入文件的路径和文件名就可以执行了。
- 第二种:
bash 1.sh
- 使用bash或sh命令执行脚本。
查看脚本执行过程
bash -x 1.sh
- 脚本执行的过程会在每个步骤前显示一个“+”加号。
查看脚本是否语法错误
bash -n 1.sh
- 执行后没有任何输出,说明没有错误。
- 无意义或不存在的命令不会被检出错误
- 语法检测只针对:语句执行的过程中,是否违背了shell语法。
20.3 date命令用法
1,显示日期与时间
- 显示当前的时间
[root@lnmp ~]# date
2019年 06月 15日 星期六 14:04:00 CST
- 如果需要显示英文可以使用以下命令切换系统语言
LANG=en
2,定制date命令显示的内容
- 显示年(四位数表示)
[root@lnmp ~]# date +%Y
2019
- 显示年(两位数表示)
[root@lnmp ~]# date +%y
19
- 显示月
[root@lnmp ~]# date +%m
06
- 显示日
[root@lnmp ~]# date +%d
15
- 显示分
[root@lnmp ~]# date +%M
05
- 显示时
[root@lnmp ~]# date +%H
14
- 显示秒
[root@lnmp ~]# date +%S
55
- 显示距1970年零点零分到现在过去了多少秒
[root@lnmp ~]# date +%s
1560578769
- 显示月份
[root@lnmp ~]# date +%h
6月
- 显示今天是星期几
[root@lnmp ~]# date +%w
6
- 显示现在是今年的第几周
[root@lnmp ~]# date +%W
23
- 如果使用
date +%Y%m%d
就会以纯数字的方式显示年月日。
比如现在是2019年6月15日。显示的结果为20190615 - 以固定方式显示当前日期或时间
date +%D
06/15/19
date +%F
2019-06-15
date +%T
13:43:58
- 显示日历
[root@lnmp ~]# cal
六月 2019
日 一 二 三 四 五 六
1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30
当前日期显示为高亮
3,参数-d的使用
- 一天后
[root@lnmp ~]# date -d "+1day"
2019年 06月 16日 星期日 14:08:04 CST
引号不能取消,但是+1和day之前允许有空格。
- 一天前
[root@lnmp ~]# date -d "-1 day"
2019年 06月 14日 星期五 14:09:48 CST
- 一月前
[root@lnmp ~]# date -d "-1 month"
2019年 05月 15日 星期三 14:11:04 CST
- 一年前
[root@lnmp ~]# date -d "-1 year"
2018年 06月 15日 星期五 14:11:34 CST
- 一分钟前
[root@lnmp ~]# date -d "-1 min"
2019年 06月 15日 星期六 14:11:07 CST
- 一天前只取年月日
[root@lnmp ~]# date -d "-1 day" +%F
2019-06-14
[root@lnmp ~]# date -d "-1 day" +%D
06/14/19
- 一小时前只取分时秒
[root@lnmp ~]# date -d "-1 hour" +%T
13:14:33
4,时间戳与时间的互换
- 已知时间戳转换为具体的时间
[root@lnmp ~]# date -d @1560579488
2019年 06月 15日 星期六 14:18:08 CST
- 具体的时间转换为时间戳
[root@lnmp ~]# date +%s -d "2019-06-15 14:18:08"
1560579488
20.4 shell脚本中的变量
为什么要使用变量
- 当脚本中使用某个字符串较频繁并且字符串长度很长时就应该使用变量代替
- 使用条件语句时,常使用变量 if [ $a -gt 1 ]; then … ; fi
这里的变量是一个动态结果 - 引用某个命令的结果时,用变量替代 a=
wc -l 1.txt
这里的变量的值就是命令的结果。结合上边的内容就可以生成一个条件判断:
当1.txt文件行数满足某个条件时,如何如何。 - 写和用户交互的脚本时,变量也是必不可少的
read -p "Input a number: " n; echo $n - 内置变量 $0, $1, $2… $0表示脚本本身,$1 第一个参数,$2 第二个 … $#显示参数个数
通常我们执行时加上start之类的参数,就是匹配了脚本内相关的语句,所以才能执行的。 - 数学运算a=1;b=2; c= ( ( (( ((a+ b ) ) 或 者 b))或者 b))或者[ a + a+ a+b]
20.5 shell脚本中的逻辑判断
格式1:if 条件 ; then 语句; fi
- 如果符合条件就执行语句,fi结束判断。
#!/bin/bash
read -p "input number :" a
if [ $a -gt 3 ]
then
echo good
fi
- 输入一个大于3的数字就会显示good.小于3则什么都不会显示
格式2:if 条件; then 语句1; else 语句2; fi
- 如果符合条件就执行语句1,如果不符合就执行语句2。
#!/bin/bash
read -p "input number :" a
if [ $a -gt 9 ]
then
echo good
else
echo bad
fi
- 输入一个数字如果大于9就显示good 小于就显示bad
格式3:if …; then … ;elif …; then …; else …; fi
#!/bin/bash
read -p "input number :" a
if [ $a -gt 90 ]
then
echo verygood
elif [ $a -gt 60 ]
then
echo good
else
echo bad
fi
- 多重条件判断。当无法满足第一条件后,才继续在结果中对比是否满足第二条件。只要满足其中一条,就输出结果。
如果都不满足,就显示else的结果。
逻辑判断表达式
- 等于
-eq
- 不等于
-ne
- 大于
-gt
- 小于
lt
- 大于等于
-ge
- 小于等于
-le
多条件判断
if [ $a -gt 5 ] && [ $a -lt 10 ]; then
if [ $b -gt 5 ] || [ $b -lt 3 ]; then
- &&表示并且。||表示或者。
20.6 if的文件目录属性判断
- [ -f file ]判断是否是普通文件,且存在
- [ -d file ] 判断是否是目录,且存在
- [ -e file ] 判断文件或目录是否存在
- [ -r file ] 判断文件是否可读
- [ -w file ] 判断文件是否可写
- [ -x file ] 判断文件是否可执行
- 判断时加“!”叹号表示取反
20.7 if特殊用法
if [ -z “$a” ]这个表示当变量a的值为空时会怎么样
- 在写脚本时,有时会遇到文件不存在(比如被别人删除了)。
- 如果在使用这个文件(脚本会把这个文件赋予一个变量)前先判断变量值是否为空。就会排除类似的BUG产生。
#!/bin/bash
n=`wc -l /tmp/lalal`
if [ -z "$n" ]
then
echo error
exit
elif [ $n -gt 100 ]
then
echo good
else
echo bad
fi
- 先判断变量值是否为空。如果为空则输出“错误”,然后退出脚本。
- 当值不为空时判断是否大于100,如果是输出good。如果否输出bad。
- 不论使用-z判断还是-n判断。变量都应该用双引号括起来。如果是文件不需要引号。
if [ -n “$a” ] 表示当变量a的值不为空
if grep -q ‘123’ 1.txt; then 表示如果1.txt中含有’123’的行时会怎么样
- grep -q仅匹配,但不显示匹配的结果。
if [ ! -e file ]; then 表示文件不存在时会怎么样
- 这里还是判断文件是否存在的取反操作。
if (($a<1)); then …等同于 if [ $a -lt 1 ]; then…
[ ]
中不能使用<,>,==,!=,>=,<=这样的符号
20.8/20.9 case判断
1,格式
case 变量名 in
变量值1)
command
;;
变量值2)
command
;;
变量值3)
command
;;
...... \\省略,中间可以安排很多匹配。
*)
command
;;
esac
- 当变量的值满足其中一项时,执行指定的内容。
- *表示除了以上所有的情况外,执行的内容。
- 在变量值的匹配过程中可以便用竖线“|”来表示或者的意思。比如
2|3)
command
;;
就表示当变量值为2或3时执行的内容。
2,shell脚本案例
#!/bin/bash
read -p "Please input a number: " n
if [ -z "$n" ]
then
echo "Please input a number."
exit 1
fi
n1=`echo $n|sed 's/[0-9]//g'`
if [ -n "$n1" ]
then
echo "Please input a number."
exit 1
fi
if [ $n -lt 60 ] && [ $n -ge 0 ]
then
tag=1
elif [ $n -ge 60 ] && [ $n -lt 80 ]
then
tag=2
elif [ $n -ge 80 ] && [ $n -lt 90 ]
then
tag=3
elif [ $n -ge 90 ] && [ $n -le 100 ]
then
tag=4
else
tag=0
fi
case $tag in
1)
echo "not ok"
;;
2)
echo "ok"
;;
3)
echo "ook"
;;
4)
echo "oook"
;;
*)
echo "The number range is 0-100."
;;
esac
3,脚本分析
- 交互模块
#!/bin/bash
read -p "Please input a number: " n
会把用户输入的内容赋值给n
- 除错模块
if [ -z "$n" ]
then
echo "Please input a number."
exit 1
fi
n1=`echo $n|sed 's/[0-9]//g'`
if [ -n "$n1" ]
then
echo "Please input a number."
exit 1
fi
先判断变量n是否为空,再判断变量是否为纯数字。
如果是空变量,或者不是纯数字则退出。
exit后的参数可以用echo $?
查看。
- 评级模块
if [ $n -lt 60 ] && [ $n -ge 0 ]
then
tag=1
elif [ $n -ge 60 ] && [ $n -lt 80 ]
then
tag=2
elif [ $n -ge 80 ] && [ $n -lt 90 ]
then
tag=3
elif [ $n -ge 90 ] && [ $n -le 100 ]
then
tag=4
else
tag=0
fi
对输入的值进行了四个级别的划分
大于等于零小于60,给tag赋值1。大于等于60,小于80,tag=2。大于等于80小于90,tag=3。大于等于90,小于等于100。tag=4。
除了以上的四种情况tag=0
- 评价模块
case $tag in
1)
echo "not ok"
;;
2)
echo "ok"
;;
3)
echo "ook"
;;
4)
echo "oook"
;;
*)
echo "The number range is 0-100."
;;
esac
就是对上边模块的四种情况进行评价。
20.10 for循环
语法:
for 变量名 in 条件; do …; done
案例1
#!/bin/bash
sum=0
for i in `seq 1 100`
do
sum=$[$sum+$i]
echo $i
done
echo $sum
- 就是依次把条件的结果赋值给变量。
- 这次使用的是“计数器”。从1数到100。
- 循环做的内容是“累加”
案例2
#!/bin/bash
cd /etc/
for a in `ls /etc/`
do
if [ -d $a ]
then
ls -d $a
fi
done
- 这里条件的结果是列出/etc/下所有的文件和目录。
- 针对每一个文件和目录进行判断。
- 当前变量为目录时时则列出他
- 遍历的对象是以空格做为分隔符。如果ls一个目录时,这个目录下的文件或目录名包含有空格。会从空格处分成两部分处理。
20.11/20.12 while循环
语法:
while 条件; do … ; done
- while 是一个死循环,会不停的重复从do到done的过程。
案例1
while :
do
load=`w|head -1|awk -F 'load average: ' '{print $2}'|cut -d. -f1`
if [ $load -gt 10 ]
then
top|mail -s "load is high: $load" asldkfls@11.com
fi
sleep 30
done
- while后边的冒号表示:“真”。
- 这是一个系统负载监控脚本。如果不想被意外中断,可以把他放到一个screen里去。
w|head -1|awk -F 'load average: ' '{print $2}'|cut -d. -f1
的目的是把负载里的每分负载内容提取出来。
案例2
#!/bin/bash
while :
do
read -p "Please input a number: " n
if [ -z "$n" ]
then
echo "you need input sth."
continue
fi
n1=`echo $n|sed 's/[0-9]//g'`
if [ -n "$n1" ]
then
echo "you just only input numbers."
continue
fi
break
done
echo $n
- 这是一个针输入内容查错的模块。
- 如果用户输入的内容为空或者不是纯数字,将重新开始要求用户输入内容。
- 直到用户输入纯数字,完成对n的赋值。中止所有循环。
continue、break和exit的区别
- 循环语句执行到continue时,会直接回到do重新开始循环。
- 循环语句执行到break,时会直接跳转到done,结束这个从do到done的所有内容。
- 脚本执行到exit时会退出整个脚本。
直播笔记
- 脚本最终的目的是批量的执行系统的命令。
- 所以只有充分掌握系统的操作才能写好脚本。