Shell脚本

Shell脚本

Shell是什么?

  • Shell脚本语言属于弱类型语言,解析用户输入的命令和程序,使得用户可以与Linux进行交互;
  • 适合处理纯文本类型数据(日志、配置文件、文本、网页文件、大多数纯文本类型的文件)。

Shell概念

shebang

  • 即文件的第一行前两个字符 #!,后面的语句指定命令的解析器。

    • #!/bin/sh 或 #!/bin/bash:执行时会调用/bin/sh,即bash解析器。
    • #!/bin/perl:执行时。
    • #!/usr/bin/pythonpython解析器。
    • #!/usr/bin/env:跨平台都能正常找到解析器的办法。
    • 注意
      • 未指定shebang时,默认用Shell解析。
      • 当指定shebang,但程序是不可执行文件时,转而用Shell解析。
      • 指定shebang解析器的路径需要绝对路径。
      • 执行时指定解析器执行,会忽略shebang解析器。

变量

  • 命名规则: 只能数字、字母、下划线,不能以数字开头,区分大小写。
  • 数据类型:bash默认所有变量都是字符串。
  • 作用域:只针对当前的Shell进程,每次调用bash/sh解析器执行脚本都会开启一个子Shell
  • 进程树
# psTree 可以检查进程树
pstree
ps -ef --forest 

常用命令

# \ 转义
echo "print each param from \"\$*\""

# 变量定义与赋值之间不得有空格
name="I am Soul!"
# 单引号不识别特别语法,双引号识别特殊符号
name2="${name}"

# 变量替换/引用
echo ${name}
echo $name

# 转义 即 变量中定义命令
name = `ls`
# 1)执行 name 变量时实际上是输出 ls 指令
echo $name

# 常用命令

set											 			# 输出所有变量(局部、全局)
unset 变量名										        # 删除变量/函数
env											 			# 输出全局变量
declare													# 输出所有变量
export													# 输出/设置环境变量
export | awk -F '[ :=]' '{print $3}'					# 输出系统环境变量关键字
readonly										        # 设置只读
let											            # 数值运算 即 (())
expr											        # 计算器
expr yn.png ":" ".*"									# 统计 yn.png 文件字符个数

test -e soul.jpg && echo "OK" || echo "NO"				# 三目运算

环境变量

  • 常用环境变量
${USER}						# 当前用户名字
$UID						# 当前用户ID
$HOME						# 当前用户文件目录
$(data)						# 时间
  • 配置文件
# 用户个人配置
~/.bash_profile		
~/.bashrc					# 远程登录用户特有文件

# 全局配置文件
/etc/profile
/etc/bashrc
/etc/profile.d/				# 系统建议创建在该目录而非直接修改全局配置文件。
  • 环境变量文件加载顺序

特殊变量

# 特殊参数变量
$0							              # 获取Shell脚本文件名,以及脚本路径
$n							              # 获取脚本的第n个参数(n>=1),当大于9时,需要写成 ${10}
$#							              # 获取执行的Shell脚本后面的参数的总数
$*							              # 获取Shell脚本所有参数,加双引号时,将所有参数视为一份数据
$@							              # 获取Shell脚本所有参数,加双引号时,仍然将每个参数视为独立的数据

# 特殊状态变量
$? 							              # 上一次执行命令的状态返回值:0:成功,否则失败。
$$							              # 当前Shell脚本的进程号
$!							              # 上一次后台进程的PID
$_							              # 获取上一次执行命令最后一个参数

扩展变量

${parameter:-word}			              # param参数为空时返回word值
${parameter:=word}			              # param参数为空时将word值返回给param参数并作为返回值
${parameter:?word}			              # param参数为空时返回word作为提示信息返回
${parameter:+word}			              # param参数为空时不做处理,否则返回word

bash的内置命令

echo -n 							      # 不换行输出
echo -e 							      # 解析字符串中的特殊字符(\n:换行,\r:回车,\t:制表符,\b:退格)
echo $name | wc -l						  # 统计有多少行
echo $name | wc -L						  # 统计长度
echo "8.8*2" | bc						  # 执行 bc 运算
eval ls;cd /tmp							  # 执行多个命令
exec								      # 不创建子进程,执行后续命令,且执行完毕后自动退出
seq								          # 生成序列指令
seq -s ":" 10							  # 生成 1~10的序列并用分号隔开
time 								      # 计算执行时间
read -t 5 -p "请输入:"					  # 提示用户,输入信息(-p),-t 超时时间

# 运算

# 1) 求和运算
echo {1..100} | tr " " "+" | bc			  # 求和运算
echo $((`seq -s "+" 100`))				  # (())
seq -s " + " 100 | xargs expr			  # xargs expr

# 2)awk方式运算
echo "3.2 2.2" | awk '{print $1+$2}'

# 测试
test 								
[  ]								      # 也可以用中括号 [  ] 代替test,前后必须有空格,变量必须用""
test -e a.txt -a -f a.txt				  # -a 即: and && 条件
test -e a.txt -o -f a.txt				  # -o 即: or || 条件

# 1) 文件相关
test -e 							      # 文件是否存在
test -f 							      # 是否为文件
test -d								      # 是否为目录
test -r								      # 文件是否有[可读]属性
test -w								      # 文件是否有[可写]属性
test -x								      # 文件是否有[可执行]属性

# 2)字符串
test -z								      # 字符串为空时为真,否则为假。
test -n 							      # 字符串为空时为假,否则为真。

# 3)比较
test -eq							      # 两个数值相等
test -nq							      # 两个数值不相等
test -gt							      # a 大于 b
test -lt							      # a 小于 b
test -ge							      # a 大于等于 b
test -le							      # a 小于等于 b
test =								      # 
test !=								      # 
test !								      # 

shell子串

${name}								      # 返回变量值
${#name}							      # 返回变量长度,字符串长度
${name:start}							  # 从第几位截取返回子字符串
${name:start:length}					  # 从第几位开始,长度为length进行截取
${name#word}							  # 从变量开头删除最短的word字符串
${name##word}							  # 从变量开头删除最长word字符串
${name%word}							  # 从变量结尾删除最短的word字符串
${name%%word}							  # 从变量结尾删除最长的word字符串
${name/pattern/string}					  # 用string替换第一个匹配的pattern
${name//pattern/string}					  # 用string替换所有的pattern

# 统计子串长度
echo ${#name}							  # 返回长度(最快)
echo $name | wc -l						  # wc 统计有多少行
echo $name | wc -L						  # wc 统计长度
expr length "${name}"					  # expr 计算数值
echo $name | awk '{print length($0)}'	  # awk 统计长度,length函数

执行方式

# 进程列表,并在子进程运行
(cd ~;pwd;ls;cd /tmp/;pwd;echo $BASH_SUBSHELL)
# 利用()括号开启子进程执行命令,$BASH_SUBSHELL检查当前进程层数(子进程大于1),常用于多进程处理提高程序并发执行的效率。


# 方式一(常用): 文件没有执行权限 或 脚本没有指定 shebang 都可以执行,会开启子进程执行Shell(获取不了当前进程的变量)
bash script.sh
sh script.sh

# 方式二:绝对/相对路径执行脚本,需要文件含有X权限。
bash /usr/script.sh
bash .script.sh

# 方式三:不会开启子进程执行Shell,可以获取当前进程的变量。
source script.sh
. script.sh

# 方式四:
sh < script.sh

Shell脚本应用

批量操作文件名

# 批量创建文件
touch soul_config{1..2}.xml
 
# 批量修改文件名
for item in `ls *config.xml`					# 获取当前目录下部分后缀的文件名
do mv $item `echo ${item//config/}`				# 将config文件名替换为空
done

数据备份并删除过期数据

# find xargs
find ${dir_path:=/data/mysql_back_data/} -name '*.tar.gz' -type f -mtime +7 | xargs rm -f
# 避免dir_path为空时取指定路径下的数据

运算脚本开发

#!/bin/bash/

# 开发一个 运算脚本

# 定义打印函数
print_soul(){
	printf "Please enter an interger!\n"
	exit 1
}

# 接收用户输入的第一个数字
read -p "Please input your number: " firstnum

# [] 中括号前后必须有空格
# -n 判断条件是否为空字符串
# sed 将所有数字替换为空
if [ -n "`echo $firstnum|sed 's/[0-9]//g'`" ]; then
	print_soul
fi

# 接收用户输入的 运算符
read -p "Please input your operator:" operator
if [ "${operator}" != "+" ] && [ "${operator}" != "-" ] && [ "${operator}" != "/" ] && [ "${operator}" != "*" ]; then
	echo "只允许 输入 + | - | * | /"
	exit 2
fi

# 接收用户输入的第二个数字
read -p "Please input your number: " sencondnum
if [ -n "`echo ${sencondnum} | sed 's/[0-9]//g'`" ]; then
	print_soul
fi

# 运算
echo "${firstnum}${operator}${sencondnum}结果是:" $((${firstnum}${operator}${sencondnum}))

Nginx存活检测

#!/bin/bash/

# 失败次数
fails=0
success=0

# 空转检查
while true 
	do 
		# 请求网站是否能打开
		wget --timeout=5 --tries=1 http://baidu.com/ -q -0 /dev/null
		# -ne 不等于
		if [ $? -ne 0 ]; then
			let fails=fails+1 
		else
			let success+=1
		fi

		# -ge 大于,网站正常
		if [ $success -ge 1 ]; then
			echo "The inter site is Ok!!!"
			exit 0
		fi

		# -ge 大于,网站异常、并发送邮件
		if [ $fails -ge 2 ]; then
			mail -s "`date +%F-%T`" 16042334@qq.com "The inter site is fails!!!"
			exit 2
		fi	
done

Mysql存活检测

#!/bin/bash/

# netstat、ss、lsof 三种方式检查Mysql是否运行、失败即发送邮件提醒。
# 邮件发送,需要配置:vim /etc/mail.rc,配置发送者的账号信息
if [ `netstat -tunlp | grep mysql | wc -l` != "1" -a `ss -tunlp | grep mysql | wc -l` -nq "1" -a `lsof -i tcp:3306 | wc -l` = "0" ];then 
	mail -s "`date +%F-%T`" 16042334@qq.com "The mysql is stopped!!!"
fi

Mysql备份

#!/bin/bash/

DATE=$(date +%F_%H-%M-%S)
HOST=localhost
USER=backup
PASS=123.com
BACKUP_DIR=/data/db_backup
DB_LIST=$(mysql -h$HOST	-u$USER -p$PASS -s -e "show databases;" 2> /dev/null | egrep -v "Database|information_schema|mysql|performance_schema|sys")

for item in $DB_LIST; do
	BACKUP_NAME=${BACKUP_DIR}/${DB}_${DATE}.sql
	if ! mysqldump -h$HOST	-u$USER -p$PASS -B $DB > ${BACKUP_NAME} 2> /dev/null;then
		echo "$BACKUP_NAME is Backup failed !!!"
	fi
done
posted @ 2023-06-07 22:10  Soul-Q  阅读(166)  评论(0编辑  收藏  举报