注:本文为 “shell” 相关文章合辑。
Linux Shell 脚本入门到实战详解
宝山的博客 已于 2023-01-01 21:57:19 修改
一、Shell 入门简介
1.1 什么是 Shell
-
Shell 脚本简介
网上有很多关于 Shell 的概念介绍,大多较为官方化。如果对 Linux 命令很熟悉,那么编写 Shell 脚本并非难事。Shell 本质上是将 Linux 命令组合在一起,实现某个目标,从而减轻工作量,提高工作效率。
-
官方化的 Shell 介绍
Shell 通过提示用户输入,向操作系统解释该输入,然后处理来自操作系统的任何结果输出。简单来说,Shell 是一个用户与操作系统之间的命令解释器。
-
常见的 Shell 有哪些
- Bourne Shell(
/usr/bin/sh
或/bin/sh
) - Bourne Again Shell(
/bin/bash
) - C Shell(
/usr/bin/csh
) - K Shell(
/usr/bin/ksh
) - Shell for Root(
/sbin/sh
)
最常用的 Shell 是 Bash(Bourne Again Shell)。由于其易用性和免费性,Bash 在日常工作中被广泛使用,也是大多数 Linux 操作系统默认的 Shell 环境。
- Bourne Shell(
1.2 Shell 编程注意事项
- Shell 命名:Shell 脚本名称通常为英文,后缀以
.sh
结尾,避免使用特殊符号和空格。 - 命名要直观,一眼能看出功能。
- Shell 脚本首行需要以
#!/bin/bash
开头。 - Shell 脚本变量不能以数字或特殊符号开头,可以使用下划线
_
,但不能使用破折号-
。
1.3 第一个 Shell 脚本:Hello World
-
创建一个伟大的编程项目 — Hello World
# 创建一个 Helloword.sh 文件 [root@aly_server01~]# touch Helloword.sh
# 编辑 Helloword.sh 文件 [root@aly_server01~]# vim Helloword.sh
[root@aly_server01~]# cat Helloword.sh #!/bin/bash # This is our first shell # by author rivers 2021.09 echo "hello world"
[root@aly_server01~]# ll Helloword.sh -rw-r--r-- 1 root root 85 Sep 20 22:26 Helloword.sh
# 赋予执行权限 [root@aly_server01~]# chmod o+x Helloword.sh
# 运行 Helloword.sh 脚本 [root@aly_server01~]# ./Helloword.sh hello world
二、Shell 环境变量讲解
2.1 Shell 变量详解
-
环境变量介绍
# 什么是变量 很多人可能会说,可以变化的量就是变量。但有时汉语的意思虽强大,却不一定能完全理解。这里可以理解为 a = 1,同时还可以 a = 2、a = 3,不同的值都可以赋给同一个变量 a。 # 常见的三种变量 Shell 编程中变量分为三种:系统变量、环境变量和用户变量。Shell 变量名在定义时,首个字符必须为字母(a-z,A-Z),不能以数字开头,中间不能有空格,可以使用下划线(_),不能使用(-),也不能使用标点符号等。 # 简单的变量介绍 [root@keeplived_server~]# a=18 [root@keeplived_server~]# echo $a 18
2.2 Shell 系统变量介绍
-
系统变量
# Shell 常见的变量之一是系统变量,主要用于参数判断和命令返回值判断。 $0 当前脚本的名称; $n 当前脚本的第 n 个参数,n = 1, 2, ..., 9; $* 当前脚本的所有参数(不包括程序本身); $# 当前脚本的参数个数(不包括程序本身); $? 命令或程序执行完后的状态,返回 0 表示执行成功; $$ 程序本身的 PID 号。
2.3 Shell 环境变量介绍
2.3.1 常见的系统环境变量
-
环境变量介绍
# Shell 常见的变量之二:环境变量,主要在程序运行时需要设置。 PATH 命令路径,以冒号为分隔; HOME 打印用户家目录; SHELL 显示当前 Shell 类型; USER 打印当前用户名; ID 打印当前用户 ID 信息; PWD 显示当前所在路径; TERM 打印当前终端类型; HOSTNAME 显示当前主机名; PS1 定义主机命令提示符; HISTSIZE 历史命令大小,可通过 HISTTIMEFORMAT 变量设置命令执行时间; RANDOM 随机生成一个 0 至 32767 的整数。
2.4 Shell 用户环境变量介绍
2.4.1 自定义 Shell 环境变量
-
用户自定义变量
# 用户变量又称为局部变量,主要用于 Shell 脚本内部或临时局部使用。 a=rivers 自定义变量 A; Httpd_sort=httpd-2.4.6-97.tar 自定义变量 N_SOFT; BACK_DIR=/data/backup/ 自定义变量 BACK_DIR; IPaddress=10.0.0.1 自定义变量 IP1;
2.4.2 使用 echo 打印菜单栏
-
使用 echo 打印菜单栏,显示 http-2.4 安装过程
# 创建 httpd_2.4 安装脚本 [root@web-server01~]# touch httpd_2.4_install.sh
# 赋予执行权限 [root@web-server01~]# chmod o+x httpd_2.4_install.sh
# 运行脚本 [root@web-server01~]# ./httpd_2.4_install.sh
2.4.3 Shell 中彩色输出 HelloWorld
-
echo -e 扩展
#!/bin/bash # This is echo color shell # by author rivers 2021.09-23 # 字体颜色 for i in { 31..37}; do echo -e "\033[$i;40mHello world!\033[0m" done # 背景颜色 for i in { 41..47}; do echo -e "\033[47;${i}mHello world!\033[0m" done # 显示方式 for i in { 1..8}; do echo -e "\033[$i;31;40mHello world!\033[0m" done
三、Shell 编程流程控制语句
3.1 if 条件语句介绍
3.1.1 常用的单/双分支
-
if 条件语句
# If 条件判断语句,通常以 if 开头,fi 结尾。也可加入 else 或 elif 进行多条件判断。 # 单分支语句 —— 比较大小 if [ 条件表达式 ]; then 语句1 fi # 双分支 if 语句 if [ 表达式 ]; then 语句1 else 语句2 fi # 多分支条件语句 —— 判断成绩 if [ 表达式 ]; then 语句1 elif [ 表达式 ]; then 语句2 else 语句3 fi
3.1.2 if 常见判断逻辑运算符详解
-
常见逻辑判断运算符
-f 判断文件是否存在,例如:if [ -f filename ]; -d 判断目录是否存在,例如:if [ -d dir ]; -eq 等于,应用于整型比较; -ne 不等于,应用于整型比较; -lt 小于,应用于整型比较; -gt 大于,应用于整型比较; -le 小于或等于,应用于整型比较; -ge 大于或等于,应用于整型比较; -a 逻辑与(and); -o 逻辑或(or); -z 判断字符串是否为空; -x 判断文件是否具有可执行权限; || 逻辑或; && 逻辑与。
3.1.3 使用单分支语句判断 crond 进程是否在运行 — 案例
-
判断 crond 服务是否运行
#!/bin/bash # this is check crond # by author rivers on 2021-9-23 # 定义一个变量名 name=crond num=$(ps -ef | grep $name | grep -vc grep) if [ $num -eq 1 ]; then echo "$num running!" else echo "$num is not running!" fi
3.1.4 判断系统目录是否存在 — 案例
-
判断系统目录是否存在
#!/bin/bash # this is check directory # by author rivers on 2021-9-27 if [ ! -d /data/rivers -a ! -d /tmp/rivers ]; then mkdir -p /data/rivers fi
3.1.5 多个条件判断学生分数等级 — 案例
-
判断学生成绩等级
#!/bin/bash # this check grade shell # by author rivers on 2021-09-27 grade=$1 if [ $grade -gt 90 ]; then echo "It's very good!" elif [ $grade -gt 70 ]; then echo "It's good!" elif [ $grade -ge 60 ]; then echo "pass" else echo "no pass" fi
3.2 for 循环语句介绍
-
for 循环语句
# 格式:for name [ [ in [ word ... ] ] ; ] do list ; done for 变量名 in 取值列表; do 语句 done
3.2.1 检查同一局域网多台主机是否存活
-
检查多台主机存活情况
#!/bin/bash # check hosts is on/off # by rivers on 2021-9-23 Network=$1 for Host in $(seq 1 254); do ping -c 1 $Network.$Host > /dev/null && result=0 || result=1 if [ "$result" == 0 ]; then echo -e "\033[32;1m$Network.$Host is up\033[0m" echo "$Network.$Host" >> /tmp/up.txt else echo -e "\033[31;1m$Network.$Host is down\033[0m" echo "$Network.$Host" >> /tmp/down.txt fi done
3.3 while 循环语句介绍
-
while 循环语句
# While 循环语句与 for 循环功能类似,主要用于对某个数据域进行循环读取、对文件进行遍历。通常用于需要循环某个文件或列表,满足循环条件会一直循环,不满足则退出循环。 # while 关联的还有一个 until 语句,它与 while 不同之处在于,当条件表达式为 false 时才循环,实际使用中比较少,这里不再讲解。 while [ 表达式 ]; do 语句 done
-
break 和 continue 语句
# break 是终止循环。 # continue 是跳出当前循环。 # 示例 1:在死循环中,满足条件终止循环 while true; do let N++ if [ $N -eq 5 ]; then break fi echo $N done # 输出:1 2 3 4 # 示例 2:说明 continue 的用法 N=0 while [ $N -lt 5 ]; do let N++ if [ $N -eq 3 ]; then continue fi echo $N done # 输出:1 2 4 # 打印 1-100 数字 i=0 while ((i <= 100)); do echo $i i=$((i + 1)) done
3.3.1 While 循环求 1-100 的总和 — 案例
-
求 1-100 的总和
#!/bin/bash # by author rivers on 2021-9-27 j=0 i=1 while ((i <= 100)); do j=$((j + i)) ((i++)) done echo $j
3.3.2 每 10 秒循环判断一次 hbs 用户是否登录系统 — 案例
-
每 10 秒循环判断系统登录
[root@web-server01~/script]# vim login.sh #!/bin/bash # Check File to change. # By author rivers 2021-9-27 USERS="hbs" while true; do echo "The Time is $(date +%F-%T)" sleep 10 NUM=$(who | grep "$USERS" | wc -l) if [[ $NUM -ge 1 ]]; then echo "The $USERS is login in system." fi done
3.4 case 选择语句介绍
-
case 选择语句
# Case 选择语句,主要用于对多个选择条件进行匹配输出,与 if elif 语句结构类似,通常用于脚本传递输入参数,打印输出结果及内容。 # 语法格式以 Case…in 开头,esac 结尾。每个模式必须以右括号结束,命令结尾以双分号结束。 case 模式名 in 模式1) 命令 ;; 模式2) 命令 ;; *) 不符合以上模式执行的命令 ;; esac
3.4.1 使用 case 编写一个 httpd 服务启动脚本
-
编写 http 服务启动脚本
[root@web-server01~/script]# vim httpd_start.sh #!/bin/bash # check http server start|stop|status # by author rivers on 2021-9-27 while true; do echo -e " \033[31m start \033[0m \033[32m stop \033[0m \033[33m status \033[0m \033[34m quit \033[0m " read -p "请输入选择(start|stop|quit):" char case $char in start) systemctl start httpd && echo "httpd服务已经开启" || echo "开启失败" ;; stop) systemctl stop httpd && echo "httpd服务已经关闭" || echo "关闭失败" ;; restart) systemctl restart httpd && echo "httpd服务已经重启" || echo "重启失败" ;; status) systemctl status httpd ;; quit) exit ;; *) echo "输入错误,请重新输入!" ;; esac done
3.5 select 选择语句介绍
-
select 选择语句
# select 是一个类似于 for 循环的语句,常用于选择菜单的创建,可以配合 PS3 打印菜单的输出信息。 # 语法格式以 select…in do 开头,done 结尾。 select i in (表达式); do 语句 done
3.5.1 使用 select 打印 lnmp 菜单栏 — 案例
-
打印 lnmp 选择菜单
#!/bin/bash # by author rivers on 2021-9-27 PS3="Please enter your select install menu:" select i in http php mysql quit; do case $i in http) echo -e "\033[31m Test Httpd \033[0m" ;; php) echo -e "\033[32m Test PHP \033[0m" ;; mysql