一、概述
Shell 是一个命令行解释器,它为用户提供了一个向 Linux 内核发送请求以便运行程序的界面系统级程序,用户可以用 Shell 来启动、挂起、停止甚至是编写一些程序!
1、第一个Shell脚本
- 脚本以
#!/bin/bash
开头:指明使用的是Bash Shell
- 脚本需要有可执行权限
vim hello.sh:
1 2
| #!/bin/bash echo "hello,world~"
|
2、Shell脚本执行
- 方式一:赋予脚本
x
可执行权限chmod u+x xx.sh
,然后直接./xxx.sh
执行
- 方式二:直接使用命令:
sh xx.sh
执行
二、Shell变量
1、概述
Linux Shell 中的变量分为,系统变量和用户自定义变量。
系统变量:$HOME、$PWD、$SHELL、$USER
等等,比如: echo $HOME
显示当前 shell 中所有变量:set
2、shell 变量的定义
定义变量及撤销变量:
- 定义变量:
变量名=值
- 撤销变量:
unset 变量
- 声明静态变量:
readonly 变量
。注意:不能 unset
将命令的返回值赋给变量:
定义变量的规则:
- 变量名称可以由字母、数字和下划线组成,但是不能以数字开头
- 等号两侧不能有空格
- 变量名称一般习惯为大写, 这是一个规范,我们遵守即可
注释:
几个例子:
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
| #!/bin/bash
A=100 echo $A echo A=$A echo "A=$A"
unset A echo A=$A
readonly B=2 echo "B=$B"
C=`date` echo $C D=$(date) echo $D
echo "tomcat_home=$TOMCAT_HOME"
:<<! 我是注释 ! echo "test commont!"
|
3、设置环境变量
这个环境变量的作用域是全局的,谁都可以调用!
环境变量配置文件地址:/etc/profile
export 变量名=变量值
:将 shell 变量输出为环境变量/全局变量
source 配置文件
:让修改后的配置信息立即生效
echo $变量名
:查询环境变量的值
例子:
1 2
| export TOMCAT_HOME=/opt/tomcat
|
4、位置参数变量
当我们执行一个 shell 脚本时,如果希望获取到命令行的参数信息,就可以使用到位置参数变量!
$n
:n 为数字,$0
代表命令本身,$1-$9
代表第一到第九个参数,十以上的参数,十以上的参数需要用大括号包含,如${10}
$*
:这个变量代表命令行中所有的参数,$*
把所有的参数看成一个整体
$@
:这个变量也代表命令行中所有的参数,不过$@
把每个参数区分对待
$#
:这个变量代表命令行中所有参数的个数
sh内容:
1 2 3 4 5
| #!/bin/bash echo "0=$0 1=$1 2=$2" echo "所有的参数=$*" echo "$@" echo "参数的个数=$#"
|
运行结果:
1 2 3 4 5
| [root@itnxd100 shell] 0=./myshell.sh 1=1 2=2 所有的参数=1 2 3 4 1 2 3 4 参数的个数=4
|
5、预定义变量
shell 设计者事先已经定义好的变量,可以直接在 shell 脚本中使用!
$$
:当前进程的进程号(PID)
$!
:后台运行的最后一个进程的进程号(PID)
$?
:最后一次执行的命令的返回状态。如果这个变量的值为 0,证明上一个命令正确执行;如果这个变量的值为非 0,则证明上一个命令执行不正确了。
sh文件:
1 2 3 4 5 6 7
| #!/bin/bash echo "当前执行的进程的pid=$$"
/home/shell/myshell.sh & echo "最后一个后台方式运行的进程pid=$!" echo "执行的结果是=$?"
|
三、运算符
"$((运算式))"
或"$[运算式]"
或者 expr m + n
,expression 表达式
- 注意 expr 运算符间要有空格, 如果希望将 expr 的结果赋给某个变量,使用反引号
expr \*,/, %
:乘(需要斜杠转义),除,取余
例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| #!/bin/bash
RES1=$(((2+3)*4)) echo "res1=$RES1"
RES2=$[(2+3)*4] echo "res2=$RES2"
TEMP=`expr 2 + 3` RES4=`expr $TEMP \* 4` echo “temp=$TEMP” echo "res4=$RES4"
SUM=$[$1+$2] echo "sum=$SUM"
|
四、条件&流程
1、条件判断
语法:
[ condition ]
(注意 condition 前后要有空格,非空返回 true,可使用$?
验证)
1 2 3
| [ hspEdu ] [ ] [ condition ] && echo OK || echo notok
|
运算符:
- = 字符串比较
- 两个整数的比较
- -lt 小于
- -le 小于等于 little equal
- -eq 等于
- -gt 大于
- -ge 大于等于
- -ne 不等于
- 按照文件权限进行判断
- -r 有读的权限
- -w 有写的权限
- -x 有执行的权限
- 按照文件类型进行判断
- -f 文件存在并且是一个常规的文件
- -e 文件存在
- -d 文件存在并是一个目录
例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| if [ "ok" = "ok" ] then echo "equal" fi
if [ 23 -ge 22 ] then echo "大于" fi
if [ -f /home/shell/aaa.txt ] then echo "存在" fi
if [ itnxd ] then echo "为假" fi
|
2、流程控制
2.1、if
语法:
1 2 3 4 5 6 7 8 9 10 11 12
| if [ 条件判断式 ] then 代码 fi 或者 , 多分支 if [ 条件判断式 ] then 代码 elif [条件判断式] then 代码 fi
|
注意:[ 条件判断式 ],中括号和条件判断式之间必须有空格
例子:
1 2 3 4 5 6 7 8 9
| #!/bin/bash
if [ $1 -ge 60 ] then echo "及格了" elif [ $1 -lt 60] then echo "不及格" fi
|
2.2、case
语法:
1 2 3 4 5 6 7 8 9 10 11 12
| case $变量名 in "值 1") 如果变量的值等于值 1,则执行程序 1 ;; "值 2") 如果变量的值等于值 2,则执行程序 2 ;; …省略其他分支… *) 如果变量的值都不是以上的值,则执行此程序 ;; esac
|
例子:
1 2 3 4 5 6 7 8 9 10 11 12 13
| #!/bin/bash
case $1 in "1") echo "周一" ;; "2") echo "周二" ;; *) echo "other..." ;; esac
|
2.3、for
语法:
1 2 3 4 5 6 7 8 9 10
| for 变量 in 值 1 值 2 值 3… do 程序/代码 done
for (( 初始值;循环控制条件;变量变化 )) do 程序/代码 done
|
例子:
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
| #!/bin/bash
for i in "$*" do echo "num is $i" done
echo "==============================="
for i in "$@" do echo "num is $i" done
SUM=0 for (( i=1; i <= $1; i++ )) do SUM=$[$SUM+$i] done
echo "总和sum=$SUM"
|
2.4、while
语法:
1 2 3 4 5
| while [ 条件判断式 ] do 程序 /代码 done 注意:while 和 [有空格,条件判断式和 [也有空格
|
例子:
1 2 3 4 5 6 7 8 9 10 11
| #!/bin/bash
SUM=0 i=0 while [ $i -le $1 ] do SUM=$[$SUM+$i] i=$[$i+1] done
echo "sum=$SUM"
|
五、read读取输入
可以处理控制台输入!
语法:
read(选项)(参数)
选项:
- -p:指定读取值时的提示符;
- -t:指定读取值时等待的时间(秒),如果没有在指定的时间内输入,就不再等待了
参数:变量,指定读取值的变量名
例子:
1 2 3 4 5 6 7
| #!/bin/bash
read -p "请输入一个数NUM1=" NUM1 echo "输入的NUM1=$NUM1"
read -t 10 -p "请输入一个数NUM2=" NUM2 echo "输入的NUM2=$NUM2"
|
六、函数
1、系统函数
basename [pathname] [suffix]
:返回完整路径最后 / 的部分,常用于获取文件名
- suffix 为后缀:如果 suffix 被指定了,basename 会将 pathname 或 string 中的 suffix 去掉
dirname 文件绝对路径
:从给定的包含绝对路径的文件名中去除文件名(非目录的部分),然后返回剩下的路径(目录的部分),常用于获取路径
例子:
1 2 3 4 5
| [root@itnxd100 shell] test.tx [root@itnxd100 shell] /home/aaa [root@itnxd100 shell]
|
2、自定义函数
语法:
1 2 3 4 5 6
| [ function ] funname[()] { Action; [return int;] } 调用直接写函数名:funname [值]
|
例子:
1 2 3 4 5 6 7 8 9 10 11 12
| #!/bin/bash
function getSum(){ SUM=$[$n1+$n2] echo "和是=$SUM" }
read -p "请输入一个数n1=" n1 read -p "请输入一个数n2=" n2
getSum $n1 $n2
|
七、案例
一个数据库备份案例:
- 每天凌晨 2:30 备份 数据库 itnxd 到 /data/backup/db
- 备份开始和备份结束能够给出相应的提示信息
- 备份后的文件要求以备份时间为文件名,并打包成 .tar.gz 的形式,比如:2021-03-12_230201.tar.gz
- 在备份的同时,检查是否有 10 天前备份的数据库文件,如果有就将其删除。
/usr/sbin/mysql_db.backup.sh代码:
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 31 32 33 34
| #!/bin/bash
BACKUP=/data/backup/db
DATETIME=$(date +%Y-%m-%d_%H%M%S) echo $DATETIME
HOST=localhost
DB_USER=root
DB_PW=xxxxx
DATABASE=itnxd
[ ! -d "${BACKUP}/${DATETIME}" ] && mkdir -p "${BACKUP}/${DATETIME}"
mysqldump -u${DB_USER} -p${DB_PW} --host=${HOST} -q -R --databases ${DATABASE} | gzip > ${BACKUP}/${DATETIME}/$DATETIME.sql.gz
cd ${BACKUP} tar -zcvf $DATETIME.tar.gz ${DATETIME}
rm -rf ${BACKUP}/${DATETIME}
find ${BACKUP} -atime +10 -name "*.tar.gz" -exec rm -rf {} \; echo "备份数据库${DATABASE}成功!"
|
使用crond添加定时任务:
1 2 3 4
| [root@VM-8-2-centos db] */5 * * * * flock -xn /tmp/stargate.lock -c '/usr/local/qcloud/stargate/admin/start.sh > /dev/null 2>&1 &'
30 2 * * * /usr/sbin/mysql_db_backup.sh
|