参考:Shell教程|菜鸟教程
shell 环境
#!/bin/bash
:脚本第一行,告诉系统使用什么解释器来执行,即使用哪一个 shell
注释
单行注释用#
多行注释用 :<<
开头,!
包裹注释内容
1
2
3
4
5
6
# 这是一个单行注释
:<<!
这是多行注释
这是多行注释
这是多行注释
!
变量
1
2
3
4
# 变量 = 号两侧不能有空格,大小写敏感
my_name = "lanhuli"
echo ${ my_name }
unset my_name # 删除变量
双引号和单引号的区别
单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;
单引号字符串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。
双引号里可以有变量
双引号里可以出现转义字符
只读变量
1
2
3
4
myUrl = "https://www.google.com"
readonly myUrl
myUrl = "https://www.runoob.com"
# 执行报错,NAME: This variable is read only.
字符串变量
1
2
3
4
5
6
7
8
9
10
11
#!/bin/bash
name = "Shell"
url = "http://1.1.1.1/"
# 拼接字符串
str1 = " ${ url }${ name } "
str2 = ${ url }${ name } # 中间不能有空格
str3 = " ${ url } : ${ name } " # 中间可以出现其它字符,包括空格
######### 输出 ########
http://1.1.1.1/Shell
http://1.1.1.1/Shell
http://1.1.1.1/:Shell
获取字符串长度
1
2
3
4
string = "abcdefg"
echo ${# string } # 变量为字符串时,${#string} 等价于 ${#string[0]}
echo ${# string [0] }
echo ${ string : 1 : 4 } # 字符串第 2 个字符开始截取 4 个字符
整型变量
运算符
关系运算符
关系运算符只支持数字,不支持字符串,除非字符串的值是数字。
假定变量 a 为 10,变量 b 为 20:
字符串运算符
假定变量 a 为 “abc”,变量 b 为 “efg”:
数组
1
2
3
4
5
6
# 创建数组
my_array =( A B "C" D)
echo "第一个元素为: ${ my_array [0] } "
echo "第二个元素为: ${ my_array [1] } "
echo "第三个元素为: ${ my_array [2] } "
echo "第四个元素为: ${ my_array [3] } "
关联数组
1
2
3
4
5
6
7
declare -A array_name # 创建关联数组
declare -A site
site[ "google" ]= "www.google.com"
site[ "runoob" ]= "www.runoob.com"
site[ "taobao" ]= "www.taobao.com"
echo ${ site [ "runoob" ] }
获取数组中所有的元素
1
2
3
4
5
6
7
declare -A site
site[ "google" ]= "www.google.com"
site[ "runoob" ]= "www.runoob.com"
site[ "taobao" ]= "www.taobao.com"
echo "数组的键为: ${ !site[*] } "
echo "数组的键为: ${ !site[@] } "
获取数组的长度
1
2
3
4
5
6
7
my_array[ 0]= A
my_array[ 1]= B
my_array[ 2]= C
my_array[ 3]= D
echo "数组元素个数为: ${# my_array [*] } "
echo "数组元素个数为: ${# my_array [@] } "
流程控制
if分支语句
1
2
3
4
5
6
7
8
a = 10
b = 20
if [ $a == $b ]
then
echo "a 等于 b"
else
echo "a 不等于 b"
fi
注意:如果 else 分支没有语句执行,就不要写这个 else。
多分支语句
1
2
3
4
5
6
7
8
9
10
11
12
13
14
a = 10
b = 20
if [ $a == $b ]
then
echo "a 等于 b"
elif [ $a -gt $b ]
then
echo "a 大于 b"
elif [ $a -lt $b ]
then
echo "a 小于 b"
else
echo "没有符合的条件"
fi
for 循环
while 循环
1
2
3
4
5
6
7
#!/bin/bash
num = 2
while (( num<100)) #数值与运算符可以没有空格,变量的使用时也可以不使用$num
do
echo " $num "
(( num = num*2))
done
shell 脚本例子
安装 docker 脚本
从自己的服务器下载 docker 离线安装包
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
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
#!/bin/bash
url = "url"
docker_version = "docker-23.0.6.tgz"
docker_compose_version = "docker-compose-2.17.0"
# 判断网络环境,优先内网
echo "正在检测网络环境,请稍等..."
ping 192.168.0.211 -c 3 & > /dev/null
if [ $? -eq 0 ]
then
echo "检测到网络环境可通内网"
url = "http://192.168.0.211/"
else
# 内网不通,检测外网
ping oss.lanhuli.top -c 3 & > /dev/null
if [ $? -eq 0 ]
then
echo "检测到网络环境可通外网"
url = "http://oss.lanhuli.top/"
fi
fi
# 安装docker
echo "正在下载docker安装包,请稍等..."
wget ${ url }${ docker_version } & > /dev/null
if [ $? -eq 0 ]
then
echo ${ docker_version } "安装包下载成功"
else
echo ${ docker_version } "安装包下载失败"
fi
tar -xf ${ docker_version }
chmod 755 ./docker/*
cp ./docker/* /usr/bin/
rm -rf docker
rm -rf ${ docker_version }
echo "docker安装完毕"
echo "docker版本:" ` docker -v`
# 安装docker-compose
echo "正在下载docker-compose"
wget ${ url }${ docker_compose_version } & > /dev/null
if [ $? -eq 0 ]
then
echo ${ docker_compose_version } "下载成功"
else
echo ${ docker_compose_version } "下载失败"
fi
chmod 755 ${ docker_compose_version }
mv ${ docker_compose_version } /usr/local/bin/docker-compose
echo "docker-compose安装完毕"
echo "docker-compose版本:" ` docker-compose -v`
# 将 docker 注册成系统服务
touch /etc/systemd/system/docker.service
cat > /etc/systemd/system/docker.service<< EOF
[Unit]
Description=Docker Application Container Engine
Documentation=https://docs.docker.com
After=network-online.target firewalld.service
Wants=network-online.target
[Service]
Type=notify
ExecStart=/usr/bin/dockerd
ExecReload=/bin/kill -s HUP $MAINPID
LimitNOFILE=infinity
LimitNPROC=infinity
TimeoutStartSec=0
Delegate=yes
KillMode=process
Restart=on-failure
StartLimitBurst=3
StartLimitInterval=60s
[Install]
WantedBy=multi-user.target
EOF
chmod +x /etc/systemd/system/docker.service
systemctl daemon-reload
echo "注册完毕,可以使用systemctl命令启动docker"
安装 frp 脚本
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
#!/bin/bash
# wget http://oss.lanhuli.top/frp_0.60.0_linux_amd64.tar.gz
wget http://192.168.0.211/frp_0.60.0_linux_amd64.tar.gz
tar -xzf frp_0.60.0_linux_amd64.tar.gz
cat > frp_0.60.0_linux_amd64/frps.toml << EOF
bindPort = 16669
auth.token = "xujiajun#7732."
EOF
mv frp_0.60.0_linux_amd64 /opt/frp_0.60.0_linux_amd64
rm frp_0.60.0_linux_amd64.tar.gz
# 注册成系统服务
touch /etc/systemd/system/frps.service
cat > /etc/systemd/system/frps.service<< EOF
[Unit]
# 服务名称,可自定义
Description = frp server
After = network.target syslog.target
Wants = network.target
[Service]
Type = simple
# 启动 frps 的命令,需修改为您的 frps 的安装路径
ExecStart = /opt/frp_0.60.0_linux_amd64/frps -c /opt/frp_0.60.0_linux_amd64/frps.toml
[Install]
WantedBy = multi-user.target
EOF
chmod +x /etc/systemd/system/frps.service
systemctl daemon-reload
监控服务是否挂掉并自动重新启动
脚本主要负责查询服务是否运行,没有运行的话就启动服务。然后将这个脚本通过 crontab 定时任务每分钟运行一次
1
2
3
4
5
6
7
8
9
10
11
12
13
#!/bin/bash
# 检测服务是否运行
result = ` systemctl status nginx | grep "active (running)" `
#echo ${result}
if [[ " $result " != "" ]]
then
echo "nginx已启动"
else
echo "nginx未启动,立即启动nginx"
systemctl restart nginx
fi
上述脚本逻辑:通过 systemctl 命令查看 nginx 的运行状态,如果是启动状态,就会有关键词 active (running)
在其中,result 就不为空,通过这个来判断 nginx 是启动还是关闭状态。如果 result 为空,说明 nginx 停了,可以使用 systemctl start nginx
命令重新启动 nginx。
把脚本通过 crontab 定时任务每隔 1 分钟运行检查一次,就可以达到目的。输入 crontab -e
命令,在文件末尾添加 */1 * * * * /root/start_nginx.sh
然后保存
注意:不一定非要使用 systemctl status nginx | grep "active (running)"
进行判断,也可以使用 ps 命令查看是否存在进程。
1
2
3
4
5
6
#!/bin/bash
# 检测frpc是否运行
if [[ ` systemctl status frps | grep "active (running)" ` = "" ]] ; then systemctl start frps; fi
# 需要将 frps 注册为 systemd 服务
其它小知识
关于双小括号使用
参考: https://blog.csdn.net/u011479200/article/details/79603385
1
2
3
4
5
6
7
8
9
10
11
12
13
14
a = 10
b = 20
if (( $a == $b ))
then
echo "a 等于 b"
elif (( $a > $b ))
then
echo "a 大于 b"
elif (( $a < $b ))
then
echo "a 小于 b"
else
echo "没有符合的条件"
fi
用[]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
a = 10
b = 20
if [ $a == $b ]
then
echo "a 等于 b"
elif [ $a -gt $b ]
then
echo "a 大于 b"
elif [ $a -lt $b ]
then
echo "a 小于 b"
else
echo "没有符合的条件"
fi
关于 == 、=和-eq
在 shell 中,=和 == 运算符都可以用于判断两个字符串、两个字符串变量是否相同,== 支持模式匹配,而= 不支持模式匹配。使用 -eq 来判断两个整数是否相等。
判断命令是否执行成功
1
2
3
4
5
6
7
8
# shell中使用符号“$?”来显示上一条命令执行的返回值,如果为0则代表执行成功,其他表示失败。
ping 192.168.0.110 -c 3 & >/dev/null
if [ $? -eq 0 ]
then
echo "能ping通"
else
echo "不能ping通"
fi