首先继续上次的内容看一下另一个案例
案例:持续检查服务器负载
uptime查看负载情况(也可以用top命令)
[root@localhost ~]# uptime22:11:26 up 7:05, 3 users, load average: 0.00, 0.00, 0.00
#!/bin/bash
#Function:持续检查服务器负载,如果负载过高则发送警告邮件
while true
doload=$(uptime | awk '{print $10}' | tr -s "," ' ')if [ $(echo "$load > 1.0" | bc) -eq 1 ];thenecho "Warning:Server load is high:$load" | mail -s "Server load warning" admin@example.comfisleep 300
done
(该代码可能执行不成功)
案例:持续监控应用日志
[root@localhost ~]# grep -i "error" /var/log/messages
Aug 26 19:30:52 localhost /usr/sbin/irqbalance[1044]: Cannot change IRQ 73 affinity: Input/output error
Aug 26 19:30:52 localhost /usr/sbin/irqbalance[1044]: Cannot change IRQ 63 affinity: Input/output error
案例:逐行读取文件
[root@localhost ~]# vim test.sh
#!/bin/bash
while read line
dousername=$(echo $line | cut -d':' -f1)time=$(echo $line | cut -d':' -f3)time=$(date -d @$[time*24*3600])echo ${username}-$time
done < /root/shadow
[root@localhost ~]# ./test.sh
root-2025年 09月 08日 星期一 08:00:00 CST
bin-2024年 12月 24日 星期二 08:00:00 CST
daemon-2024年 12月 24日 星期二 08:00:00 CST
adm-2024年 12月 24日 星期二 08:00:00 CST
lp-2024年 12月 24日 星期二 08:00:00 CST
sync-2024年 12月 24日 星期二 08:00:00 CST
shutdown-2024年 12月 24日 星期二 08:00:00 CST
halt-2024年 12月 24日 星期二 08:00:00 CST
mail-2024年 12月 24日 星期二 08:00:00 CST
operator-2024年 12月 24日 星期二 08:00:00 CST
games-2024年 12月 24日 星期二 08:00:00 CST
ftp-2024年 12月 24日 星期二 08:00:00 CST
nobody-2024年 12月 24日 星期二 08:00:00 CST
systemd-coredump-2025年 08月 26日 星期二 08:00:00 CST
dbus-2025年 08月 26日 星期二 08:00:00 CST
polkitd-2025年 08月 26日 星期二 08:00:00 CST
saslauth-2025年 08月 26日 星期二 08:00:00 CST
dhcpd-2025年 08月 26日 星期二 08:00:00 CST
sshd-2025年 08月 26日 星期二 08:00:00 CST
rpc-2025年 08月 26日 星期二 08:00:00 CST
tss-2025年 08月 26日 星期二 08:00:00 CST
apache-2025年 08月 26日 星期二 08:00:00 CST
unbound-2025年 08月 26日 星期二 08:00:00 CST
libstoragemgmt-2025年 08月 26日 星期二 08:00:00 CST
cockpit-ws-2025年 08月 26日 星期二 08:00:00 CST
cockpit-wsinstance-2025年 08月 26日 星期二 08:00:00 CST
pesign-2025年 08月 26日 星期二 08:00:00 CST
chrony-2025年 08月 26日 星期二 08:00:00 CST
tcpdump-2025年 08月 26日 星期二 08:00:00 CST
tom-2025年 09月 08日 星期一 08:00:00 CST
lynn-2025年 09月 08日 星期一 08:00:00 CST
user1-2025年 09月 08日 星期一 08:00:00 CST
循环控制语句continue
continue [N]:提前结束第N层的本轮循环,而直接进入下一轮判断;最内层为第1层
用法示例:
[root@localhost ~]# vim continue.sh
#!/bin/bash
i=1
while [ $i -le 10 ]
doecho $ilet i++
done
[root@localhost ~]# chmod +x continue.sh
[root@localhost ~]# ./continue.sh
1
2
3
4
5
6
7
8
9
10
循环控制语句break
break [N]:提前结束第N层后的全部循环;最内层为第1层,默认为1
[root@localhost ~]# vim break.sh
for((i=0;i<10;i++));do[ $i -eq 5 ] && breakfor((j=0;j<10;j++));do[ $j -eq 5 ] && breakecho $jdoneecho ----------------------------
done
[root@localhost ~]# chmod +x break.sh
[root@localhost ~]# ./break.sh
0
1
2
3
4
----------------------------
0
1
2
3
4
----------------------------
0
1
2
3
4
----------------------------
0
1
2
3
4
----------------------------
0
1
2
3
4
----------------------------
shift技术
shift命令用于对参数的向左移动,通常用于在不知道传入参数个数的情况下依次遍历每个参数,然后进行相应的处理(常见与Linux中各种程序的启动脚本)。在扫描处理脚本程序的参数时,经常要用到shift命令
shift命令每执行一次,参数序列顺次左移一个位置,$#
的值减1,用于分别处理每个参数,移出去的参数不可再用
注意:$#
表示脚本后跟随的参数总的个数,$n可以获取脚本后跟随的第n个参数的值
示例:
[root@localhost ~]# vim shift.sh
#!/bin/bashwhile [ $# -ne 0 ]
doecho "一共$#个参数,内容分别是:"for i in `seq $#`;doecho -n -e "$i\t"doneechoshift
done
[root@localhost ~]# ./shift.sh 1 2
一共2个参数,内容分别是:
1 2
一共1个参数,内容分别是:
1
案例:累加1-100
[root@localhost ~]# vim sum.sh
#!/bin/bash
sum=0
for((i=0;i<=100;i++))
dolet sum+=$i
done
echo $sum
[root@localhost ~]# ./sum.sh
5050
Shell中的数组
1.(1)Shell数组的定义
数组的基本定义
在Shell
中,用小括号()
来表示数组,数组元素之间用空格来分隔
[root@localhost ~]# array1=(1 2 3 4 5 6)
[root@localhost ~]# echo ${array1[*]} #输出数组的所有元素
1 2 3 4 5 6
[root@localhost ~]# echo ${array1[@]} #输出数组的所有元素
1 2 3 4 5 6
[root@localhost ~]# echo ${array1[0]} #输出数组的第N个数(由0代表第一个数开始)
1
[root@localhost ~]# echo ${array1[2]} #输出数组的第3个数
3
(2)采用键值对的形式赋值
在Shell
中用小括号将变量括起来,同时采用键值对的形式赋值
[root@localhost ~]# array2=([1]=one [2]=two [3]=three)
[root@localhost ~]# echo ${array2[*]} #输出数组的所有元素
one two three
[root@localhost ~]# echo ${array2[@]} #输出数组的所有元素
one two three
[root@localhost ~]# echo ${#array2[@]} #输出数组的元素个数
3
(3)动态定义数组数量
动态地定义数组变量,并使用命令的输出结果作为数组的内容
[root@localhost ~]# array3=($(ls /boot))
[root@localhost ~]# echo ${#array3[*]} #输出定义数组的元素个数
13
[root@localhost ~]# echo ${array3[0]}
config-6.6.0-72.0.0.76.oe2403sp1.x86_64
2.Shell数组的打印
打印单个数组元素: ${数组名[下标]} 。当未指定数组下标时,下标默认从0开始
打印全部数组内容:${数组名[@]}或 ${数组名[*]}
打印数组元素的个数:${#数组名[@]}或 ${#数组名[*]}
3.Shell数组的赋值
如果下标不存在,则自动添加一个新的元素;如果下标存在,则覆盖原来的值
4.Shell数组的拼接合并
所谓Shell
数组拼接(数组合并),就是将两个数组连接成一个数组
拼接数组的思路是:先利用@
或者*
,将数组扩展成列表,然后再合并到一起,具体格式如下:
[root@localhost ~]# array1=(1 2 3 4 5 6)
[root@localhost ~]# array2=(7 8 9 10 11 12)
[root@localhost ~]# array3=(${array1[@]} ${array2[@]})
[root@localhost ~]# echo ${array3[*]}
1 2 3 4 5 6 7 8 9 10 11 12
5.Shell删除数组元素
在Shell
中,使用unset关键字来删除数组元素,具体格式如下:
[root@localhost ~]# unset array3[2]
[root@localhost ~]# echo ${array3[*]}
1 2 4 5 6 7 8 9 10 11 12
[root@localhost ~]# unset array3
[root@localhost ~]# echo ${array3[*]}[root@localhost ~]#
6.获取数组某范围的元素
在Shell
中直接通过${数组名[@/*]:起始位置:长度}
获取数组给定范围内元素,返回字符串,中间用空格分开
[root@localhost ~]# arr1=(1 2 3 4 5)
[root@localhost ~]# echo ${arr1[*]:1:2}
2 3
[root@localhost ~]# echo ${arr1[@]:0:4}
1 2 3 4
mapfile命令
mapfile直接用一个指令交互式的生成一个数组
[root@localhost ~]# ls /boot/ > boot.txt
[root@localhost ~]# mapfile arr6 < boot.txt
[root@localhost ~]# echo ${arr6[*]}
config-6.6.0-72.0.0.76.oe2403sp1.x86_64 dracut efi grub2 initramfs-0-rescue-5a44ea1f344449c0863349ae24820f72.img initramfs-6.6.0-72.0.0.76.oe2403sp1.x86_64.img initramfs-6.6.0-72.0.0.76.oe2403sp1.x86_64kdump.img loader lost+found symvers-6.6.0-72.0.0.76.oe2403sp1.x86_64.gz System.map-6.6.0-72.0.0.76.oe2403sp1.x86_64 vmlinuz-0-rescue-5a44ea1f344449c0863349ae24820f72 vmlinuz-6.6.0-72.0.0.76.oe2403sp1.x86_64
shell中的函数
shell函数的定义
注意:在shell里不支持函数的参数传入,只支持脚本的参数传入,所以name()内不能写任何东西。
本质的作用就是;将一段重复的代码编入脚本中,放在指定位置,使用时直接调用。
示例
[root@localhost ~]# vim sum.sh
#!/bin/bash
c1=$1
c2=$2
c3=$3
sum() {
sum1=0
for i in $(seq $c1 $c2 $c3)
dolet sum1+=$i
done
echo $sum1
}
sum
[root@localhost ~]# chmod +x sum.sh
[root@localhost ~]# ./sum.sh 1 100
5050
案例;在子shell里调用的函数在父shell里不能输出
[root@localhost ~]# vim sum.sh
#!/bin/bash
c1=$1
c2=$2
c3=$3
sum() {
sum1=0
for i in $(seq $c1 $c2 $c3)
dolet sum1+=$i
done
return $sum1
}
sum
[root@localhost ~]# ./sum.sh 1 100
[root@localhost ~]# echo $sum1[root@localhost ~]# source sum.sh 1 100
[root@localhost ~]# echo $sum1
5050
return的效果:把结果存储在一个位置上
Shell函数的调用
调用 Shell 函数时可以给它传递参数,也可以不传递。如果不传递参数,直接给出函数名字即可
Shell的八大扩展功能(不重要)
花括号
在shell脚本中,可以使用括号对字符串进行扩展,我们可以在一对花括号中包含一组以分号分隔的字符串或者字符串序列组成一个字符串扩展,注意最终输出结果以空格分隔,使用该扩展花括号不可以被引号引用,花括号的数量必须是偶数个
[root@localhost ~]# echo {1,5} #对字符串进行扩展
1 5
[root@localhost ~]# echo {hello,world} #对字符串进行扩展
hello world
[root@localhost ~]# echo {a..z} #对字符串序列进行扩展
a b c d e f g h i j k l m n o p q r s t u v w x y z
#字符串后面可以跟一个步长整数,默认为1或-1
[root@localhost ~]# echo {a..z..2}
a c e g i k m o q s u w y
[root@localhost ~]# echo {a..z..3}
a d g j m p s v y
[root@localhost ~]# echo {1..9..3}
1 4 7
[root@localhost ~]# echo {1..9..2}
1 3 5 7 9
[root@localhost ~]# echo "{a..z}" #花括号扩展不能使用引号
{a..z}
[root@localhost ~]# echo t{i,o}p #花括号前后都可以添加可选字符串
tip top
[root@localhost ~]# echo t{o,e{a,m}}p #花括号支持嵌套
top teap temp
#花括号批量操作
[root@localhost ~]# mkdir -p t{o,e{a,m}}p
[root@localhost ~]# touch t{o,e{a,m}}p/{a,b,c,d}e.txt
波浪号
波浪号在Shell脚本中默认代表当前用户家目录
[root@localhost /]# echo ~ #显示当前用户的家目录 /root [root@localhost /]# echo ~/elk /root/elk [root@localhost /]# echo ~elk #显示特定用户的家目录,该用户必须存在 /home/elk [root@localhost /]# echo ~+ #显示当前工作目录 / [root@localhost /]# echo ~- #显示前一个工作目录 /root
变量替换
在Shell脚本中我们会使用 对变量进行扩展替换,变量字符可以放到花括号中,这样可以防止需要扩展的变量字符与其他不需要扩展的字符混淆,如果 对变量进行扩展替换,变量字符可以放到花括号中,这样可以防止需要扩展的变量字符与其他不需要扩展的字符混淆,如果 对变量进行扩展替换,变量字符可以放到花括号中,这样可以防止需要扩展的变量字符与其他不需要扩展的字符混淆,如果后面是位置变量且多余一个数字,必须使用{}
[root@localhost ~]# a="hello word" [root@localhost ~]# echo $a hello word [root@localhost ~]# echo ${a} hello word [root@localhost ~]# b=a [root@localhost ~]# echo ${b} #直接返回变量的值 a [root@localhost ~]# echo ${!b} #间接引用a变量的值 hello word [root@localhost ~]# c=b [root@localhost ~]# echo ${!c} #尽可以实现一层简介引用 a
变量替换操作还可以测试变量是否存在及是否为空,若变量不存在或为空,则可以为变量设置一个默认值
Shell脚本支持多种形式的变量测试与替换功能,如下表所示
语法格式 | 功能描述 |
---|---|
${变量:-关键字} | 如果变量未定义或为空,则返回关键字,否则返回变量值 |
${变量:=关键字} | 如果变量未定义或为空,则将关键字赋值给变量,并返回结果,否则直接返回变量值 |
${变量:?关键字} | 如果变量未定义或为空,则通过标准错误显示包含关键字的错误信息,否则返回变量值 |
${变量:+关键字} | 如果变量未定义或为空,则直接返回空,否则返回关键字 |
唯一要了解basename和dirname,其余的都不重要。
basename
和dirname
:
basename
:可以获取一个路径中的文件名
dirname
:仅保留路径,删除文件名