22 shell编程
目录
一、shell编程是个啥
Shell
是一个命令行解释器,它为用户提供了一个向 Linux 内核发送请求以便运行程序的系统级程序
二、shell编程打印hello world
# 首先编辑文件
vim helloworld.sh
#!/bin/bash # 这一句是shell语言的规范,必须写(告诉计算机,使用bash解释器来执行代码)
echo 'hellp world' # echo控制台输出
三、执行代码
1. 方式一
给脚本可执行权限
chmod 744 helloworld.sh
然后直接运行脚本
2. 方式二
直接调用shell解释器执行
sh myshell.sh
四、注释
1.单行注释
#内容
2.多行注释
:<<!
内容
!
五、变量
1.变量的介绍
1.Linux中变量的分类:系统变量 自定义变量
2.系统变量:
$PATH
$HOME
$PWD
$SHELL
$USER
3.显示当前shell中所有的变量:set
2.变量的定义
基本语法
1.定义变量:变量名=变量值 (=两边不能加空格)
2.撤销变量:unset 变量名
3.声明静态变量(只能读):readonly 变量名. 静态变量不能unset
#~/bin/bash
name='chenyuxing'
age=18
echo '---------定义变量时,等号两边不能有空格-------------'
echo $name
echo $age
echo '----------------------------------------------------'
:<<!
此处是错误的,定义变量时,等号两边不能有空格
new_name = 'cecilia chen'
new_age = 18
echo `$new_name`
echo `$new_age`
!
3.变量的使用
1.定义变量name为chenyuxing
#!/bin/bash
name='chenyuxing'
2.撤销变量name
unset name
echo $name #(使用变量的时候要加$)
3.声明静态变量age=18,尝试unset撤销
readonly age=18
unset age
echo $age #(虽然我们使用了unset,但是我们发现并没有任何作用)
# unset不能撤销静态变量
4.定义规则
1.变量名称可以由字母、数字和下划线组成,但是不能以数字开头
2.等号两侧不能有空格
3.变量名称一般习惯为大写
5. 将命令的返回值赋给变量
1.A=`ls -la` 反引号,运行里面的命令,并把结果返回给变量 A
res=`ls -l /home`
echo "$res"
2.A=$(ls -la) 等价于反引号
res1=$(date)
echo "$res1"
六、设置环境变量
1.基本语法
1.export 变量名=变量值 将shell变量输出给环境变量
2.source 配置文件 让修改后的配置信息立即生效
3.echo $变量值 查看环境变量的值
#sysbianliang.sh
#~/bin/bash
#方式一
NAME="xiahong"
export NAME
#方式二
export NAME1="xiaoming"
exit
:wq
sh sysbianliang.sh (此处是执行成功的)
echo "$NAME"
echo "$NAME1"
#(此处我们发现好像没有用,因为我们没有让这个环境变量生效)
source sysbianliang.sh (让环境变量生效)
echo "$NAME"
echo "$NAME1"
#(此处可以正常打印了)
2. 快速入门
1.在/etc/profile文件中定义MY_NAME环境变量
2.查看环境变量MY_NAME的值
MY_NAME=$PATH:MY_NAME='xiaohong'
export MY_NAME
强调:在使用MY_NAME前,需要让其生效
3.source /etc/profile
4,在另外一个shell程序中使用MY_NAME
七、位置参数
1.介绍
当我们执行一个 shell脚本时,如果希望获取到命令行的参数信息,就可以使用到位置参数变量
2. 基本语法
1.
$n (功能描述:n 为数字,$0 代表命令本身,$1-$9 代表第一到第九个参数,十以上的参数,十以上的参数需要用大括号包含,如${10})
2.
$* (功能描述:这个变量代表命令行中所有的参数,$*把所有的参数看成一个整体)
3.
$@ (功能描述:这个变量也代表命令行中所有的参数,不过$@把每个参数区分对待)
4.
$#(功能描述:这个变量代表命令行中所有参数的个数)
编写一个shell脚本,pasition.sh,在脚本中获取到命令行的各个参数信息
#!/bin/bash
# 变量name age 接收在执行shell脚本时传的位置参数
name=$1
age=$2
echo "$name"
echo "$age"
#运行shell脚本
sh pasition.sh chenyuxing 18
chenyuxing
18
八、预定义变量
1. 变量
就是 shell 设计者事先已经定义好的变量,可以直接在shell 脚本中使用
2. 基本语法
$$ (功能描述:当前进程的进程号(PID))
$! (功能描述:后台运行的最后一个进程的进程号(PID))
$? (功能描述:最后一次执行的命令的返回状态。如果这个变量的值为 0,证明上一个命令正确执行;如果这个变量的值为非 0(具体是哪个数,由命令自己来决定),则证明上一个命令执行失败)
九、运算符
1.基本语法
1.$((运算式))
2.$[运算式]
3.`expr m + n`
特点:运算符之间要有空格
+
-
/
%
\*
\( \)
写一个yunsuan.sh完成:
1.3种方式计算(2+3)*4的值
#~/bin/bash
num=$(((2+3)*4))
echo $num
num1=$[(2+3)*4]
echo $num1
num2=`expr \( 2 + 3 \) \* 4`
echo $num2
2.方式2求出命令行两个参数的和
num3=$[$1+$2]
echo $num3
九、判断
1.基本语法
[ 条件 ] 注意:条件前后要有空格
特别的:
[ 非空 ] 为true
[] 为false
[ haha ] && echo true || echo false
2. 判断语句
字符串比较
= 判等
!= 判不相等
整数比较
-lt 小于
-le 小于等于
-gt 大于
-ge 大于等于
-eg 等于
-ne 不等于
文件权限判断
-r 有读的权限 [ -r 文件 ]
-w 有写的权限
-x 有执行权限
文件类型判断
-f 存在并且是一般文件 [-f 文件]
-e 文件存在
-d 存在并且是一个目录
3.案例
1.'ok'是否等于'ok'
[ 'ok'='ok' ]&& echo true || echo false
2.'ok100' 是否等于 'ok'
[ 'ok100'='ok' ]&& echo true || echo false
3.23 是否大于 23
[ 23 -gt 23 ]&& echo true || echo false
4.23 是否大于等于 23
[ 23 -ge 23 ]&& echo true || echo false
5./root是否存在
[ -d /root ]&& echo true || echo false
6./root是否是一般文件
[ -f /root ]&& echo true || echo false
十、流程控制
1. if 判断
基本语法
if [ 条件 ]
then
代码
fi
if [ 条件 ]
then
代码
else
代码
fi
if [ 条件 ]
then
代码
elif [ 条件 ]
then
代码
else
代码
fi
案列
编写shell脚本,if.sh:
如果输入参数,大于等于60,则输出'及格了',如果小于60,则输出'不及格'
#!/bin/bash
score=$1
if [ $1 -ge 90 ]
then
echo "优秀"
elif [ $1 -ge 60 ]
then
echo "及格"
else
echo "不及格"
fi
2. case选择分支
基本语法(相当诡异,令人发指,what a fuck)
case $变量名 in
'值1')
代码
;;
'值2')
代码
;;
*)
代码 都没命中执行
;;
esac
案列
编写shell脚本,case.sh:
当命令行参数是1时,输出'周一';是2时,输出'周二',是3时,输出'周三',其它情况,输出'其它'
#!/bin/bash
case $1 in 1)
echo "星期一";;
2)
echo "星期二";;
3)
echo "星期三";;
4)
echo "星期四";;
5)
echo "星期五";;
6)
echo "星期六";;
7)
echo "星期日";;
esac
3、for循环
基本语法1
for 变量 in 值1 值2 值3
do
代码
done
案例
编写foreach.sh:打印命令行输入的参数[这里可以看出$*和$@的区别]
# $*
#!/bin/bash
for i in $*
do
echo $i
done
sh for.sh 1 2 3 4 #(打印出来的是一次打印12345)
# $@
#!/bin/bash
for i in $*
do
echo $i
done
sh for.sh 1 2 3 4 #(打印4次,分别打印)
基本语法2
for ((初始值;循环条件;变量变化))
do
代码
done
编写for.sh:
从1加到100,并输出结果
#!/bin/bash
num=0
for ((i=1;i<=100;i++))
do
num=$[$num+$i]
done
echo "$num"
3. while循环
基本语法
while [ 条件 ]
do
代码
done
快速入门
编写while.sh:
从命令行中输出一个数n,统计1+...+n的值是多少
#!/bin/bash
i=1
num=0
while [ $i -le $1 ]
do
num=$[$num+$i]
i=$[$i+1]
done
echo $num
十一、与用户交互
1.基本语法
read 选项 变量
选项:
-p:提示信息
-t:等待输入的时间
2. 案列
编写input.sh:
1.读取控制带输入的值
2.读取控制台输入的值,等待6秒
十二、自定义函数
1. 基本语法
function 函数名(){
代码;
#参数使用:$1,$2,...,${10}...
return xxx;
}
调用:
函数名 值1 值2
2.案例
编写func.sh:
用函数的形式,计算两个参数的和
#!/bin/bash
function num(){
res=$[$1+$2]
echo $res
}
res=`num $1 $2`
echo $res
十三、综合案例
在/root下编写mysql_db_backuo.sh
需求:
1.每天凌晨2点10分,备份数据库mydb 到/data/backup/db
2.备份开始和备份结束时能够给出提示信息
3.备份后的文件要求以备份时间为文件名,并打包成.tar.gz的形式,如2019-09-28-044403.tar.gz
4.在备份的同时,检查是否有10天前的备份文件,如果有就删除
#!/bin/bash
#备份的路径
BACKUP=/data/backup/db
#当前的时间作为文件名
DATETIME=$(date +%Y_%m_%d_%H%M%S)
echo "=======开始备份======"
echo "=====备份的路径是 $BACKUP/$DATETIME.tar.gz"
#主机
HOST=localhost
#用户名
DB_USER=root
#密码
DB_PWD=997997
#备份的数据库
DATABASE=mydb
如果备份路径不存在,就创建
[ ! -d "$BACKUP/DATETIME" ] && mkdir -p "$BACKUP/$DATETIME"
#执行mysql的备份指令
mysqldump -u$DB_USER -p$DB_PWD --host=$HOST $DATABASE | gzip > $BACKUO/$DATETIME/$DATETIME.sql.gz
#打包备份文件
cd $BACKUP
tar -zcvf $DATETIME.tar.gz $DATETIME
#删除临时目录
rm -rf $BACKUP/$DATETIME
#删除10天前的文件
find $BACKUP -mtime +10 --name "*.tar.gz" -exec rm -rf {} \;
echo "=====备份成功+++++"