1) 如何向脚本传递参数 ?
./script argument
例子 : 显示文件名称脚本
./show.sh file1.txt
cat show.sh
#!/bin/bashe
cho $1
(LCTT 译注:谢谢某匿名访客的提醒,原题有误,修改之。)
2) 如何在脚本中使用参数 ?
第一个参数 : $1
,第二个参数 : $2
例子 : 脚本会复制文件(arg1) 到目标地址(arg2)
./copy.sh file1.txt /tmp/
cat copy.sh
#!/bin/bash
cp $1 $2
3) 如何计算传递进来的参数 ?
$#
4) 如何在脚本中获取脚本名称 ?
$0
5) 如何检查之前的命令是否运行成功 ?
$?
6) 如何获取文件的最后一行 ?
tail -1
7) 如何获取文件的第一行 ?
head -1
8) 如何获取一个文件每一行的第三个元素 ?
awk '{print $3}'
9) 假如文件中每行第一个元素是 FIND,如何获取第二个元素
awk '{ if ($1 == "FIND") print $2}'
10) 如何调试 bash 脚本
将 -xv
参数加到 #!/bin/bash
后
例子:
#!/bin/bash –xv
11) 举例如何写一个函数 ?
function example() {
echo "Hello world!"
}
12) 如何向连接两个字符串 ?
V1="Hello"
V2="World"
V3=${V1}${V2}
echo $V3
输出
HelloWorld
13) 如何进行两个整数相加 ?
V1=1
V2=2
let V3=$V1+$V2
echo $V3
输出
3
据 @kashu 的意见,本题的更佳回答为:
两个整数相加,还有若干种方法实现:
A=5 B=6 echo $(($A+$B)) # 方法 2 echo $[$A+$B] # 方法 3 expr $A + $B # 方法 4 echo $A+$B | bc # 方法 5 awk 'BEGIN{print '"$A"'+'"$B"'}' # 方法 6
14) 如何检查文件系统中是否存在某个文件 ?
if [ -f /var/log/messages ]
then
echo "File exists"
fi
15) 写出 shell 脚本中所有循环语法 ?
for 循环 :
for i in $( ls ); do
echo item: $i
done
while 循环 :
#!/bin/bash
COUNTER=0
while [ $COUNTER -lt 10 ]; do
echo The counter is $COUNTER
let COUNTER=COUNTER+1
done
until 循环 :
#!/bin/bash
COUNTER=20
until [ $COUNTER -lt 10 ]; do
echo COUNTER $COUNTER
let COUNTER-=1
done
16) 每个脚本开始的 #!/bin/sh 或 #!/bin/bash 表示什么意思 ?
这一行说明要使用的 shell。#!/bin/bash
表示脚本使用 /bin/bash
。对于 python 脚本,就是 #!/usr/bin/python
。(LCTT译注:这一行称之为释伴行。)
17) 如何获取文本文件的第 10 行 ?
head -10 file|tail -1
18) bash 脚本文件的第一个符号是什么
#
19) 命令:[ -z "" ] && echo 0 || echo 1 的输出是什么
0
20) 命令 “export” 有什么用 ?
使变量在子 shell 中可用。
21) 如何在后台运行脚本 ?
在脚本后面添加 “&”。
据 @kashu 的意见,更好的答案是:
nohup command &
大部分时间我们可能是远程使用Linux,我碰到过由于网络断线使得在后台运行的command &没了...
22) "chmod 500 script" 做什么 ?
使脚本所有者拥有可执行权限。
23) ">" 做什么 ?
重定向输出流到文件或另一个流。
24) & 和 && 有什么区别
&
- 希望脚本在后台运行的时候使用它&&
- 当前一个脚本成功完成才执行后面的命令/脚本的时候使用它
25) 什么时候要在 [ condition ] 之前使用 “if” ?
当条件满足时需要运行多条命令的时候。
26) 命令: name=John && echo 'My name is $name' 的输出是什么
My name is $name
27) bash shell 脚本中哪个符号用于注释 ?
#
28) 命令: echo ${new:-variable} 的输出是什么
variable
29) ' 和 " 引号有什么区别 ?
'
- 当我们不希望把变量转换为值的时候使用它。"
- 会计算所有变量的值并用值代替。
30) 如何在脚本文件中重定向标准输出和标准错误流到 log.txt 文件 ?
在脚本文件中添加 "exec >log.txt 2>&1"
命令。
31) 如何只用 echo 命令获取字符串变量的一部分 ?
echo ${variable:x:y}
x - 起始位置
y - 长度
例子:
variable="My name is Petras, and I am developer."
echo ${variable:11:6} # 会显示 Petras
32) 如果给定字符串 variable="User:123:321:/home/dir",如何只用 echo 命令获取 home_dir ?
echo ${variable#*:*:*:}
或
echo ${variable##*:}
33) 如何从上面的字符串中获取 “User” ?
echo ${variable%:*:*:*}
或
echo ${variable%%:*}
34) 如何使用 awk 列出 UID 小于 100 的用户 ?
awk -F: '$3<100' /etc/passwd
35) 写程序为用户计算主组数目并显示次数和组名
cat /etc/passwd|cut -d: -f4|sort|uniq -c|while read c g
do
{ echo $c; grep :$g: /etc/group|cut -d: -f1;}|xargs -n 2
done
36) 如何在 bash shell 中更改标准的域分隔符为 ":" ?
IFS=":"
37) 如何获取变量长度 ?
${#variable}
38) 如何打印变量的最后 5 个字符 ?
echo ${variable: -5}
39) ${variable:-10} 和 ${variable: -10} 有什么区别?
${variable:-10}
- 如果之前没有给variable
赋值则输出10
;如果有赋值则输出该变量${variable: -10}
- 输出variable
的最后 10 个字符
40) 如何只用 echo 命令替换字符串的一部分 ?
echo ${variable//pattern/replacement}
41) 哪个命令将命令替换为大写 ?
tr '[:lower:]' '[:upper:]'
42) 如何计算本地用户数目 ?
wc -l /etc/passwd|cut -d" " -f1
或者 cat /etc/passwd|wc -l
43) 不用 wc 命令如何计算字符串中的单词数目 ?
set ${string}
echo $#
44) "export $variable" 或 "export variable" 哪个正确 ?
export variable
45) 如何列出第二个字母是 a 或 b 的文件 ?
ls -d ?[ab]*
46) 如何将整数 a 加到 b 并赋值给 c ?
c=$((a+b))
或
c=`expr $a + $b`
或
c=`echo "$a+$b"|bc`
47) 如何去除字符串中的所有空格 ?
echo $string|tr -d " "
48) 重写这个命令,将输出变量转换为复数: item="car"; echo "I like $item" ?
item="car"; echo "I like ${item}s"
49) 写出输出数字 0 到 100 中 3 的倍数(0 3 6 9 …)的命令 ?
for i in {0..100..3}; do echo $i; done
或
for (( i=0; i<=100; i=i+3 )); do echo "Welcome $i times"; done
50) 如何打印传递给脚本的所有参数 ?
echo $*
或
echo $@
51) [ $a == $b ] 和 [ $a -eq $b ] 有什么区别
[ $a == $b ]
- 用于字符串比较[ $a -eq $b ]
- 用于数字比较
52) = 和 == 有什么区别
=
- 用于为变量赋值==
- 用于字符串比较
53) 写出测试 $a 是否大于 12 的命令 ?
[ $a -gt 12 ]
54) 写出测试 $b 是否小于等于 12 的命令 ?
[ $b -le 12 ]
55) 如何检查字符串是否以字母 "abc" 开头 ?
[[ $string == abc* ]]
56) [[ $string == abc* ]] 和 [[ $string == "abc*" ]] 有什么区别
[[ $string == abc* ]]
- 检查字符串是否以字母abc
开头[[ $string == "abc*" ]]
- 检查字符串是否完全等于abc*
57) 如何列出以 ab 或 xy 开头的用户名 ?
egrep "^ab|^xy" /etc/passwd|cut -d: -f1
58) bash 中 $! 表示什么意思 ?
后台最近执行命令的 PID.
59) $? 表示什么意思 ?
前台最近命令的结束状态。
60) 如何输出当前 shell 的 PID ?
echo $$
61) 如何获取传递给脚本的参数数目 ?
echo $#
(LCTT 译注:和第3题重复了。)
62) $* 和 $@ 有什么区别
$*
- 以一个字符串形式输出所有传递到脚本的参数$@
- 以 $IFS 为分隔符列出所有传递到脚本中的参数
63) 如何在 bash 中定义数组 ?
array=("Hi" "my" "name" "is")
64) 如何打印数组的第一个元素 ?
echo ${array[0]}
65) 如何打印数组的所有元素 ?
echo ${array[@]}
66) 如何输出所有数组索引 ?
echo ${!array[@]}
67) 如何移除数组中索引为 2 的元素 ?
unset array[2]
68) 如何在数组中添加 id 为 333 的元素 ?
array[333]="New_element"
69) shell 脚本如何获取输入的值 ?
a) 通过参数
./script param1 param2
b) 通过 read 命令
read -p "Destination backup Server : " desthost
70) 在脚本中如何使用 "expect" ?
/usr/bin/expect << EODspawn rsync -ar ${line} ${desthost}:${destpath}expect "*?assword:*"send "${password}\r"expect eofEOD
《脚本编程与Linux命令》由腾讯高级工程师 luaruan(阮永顺) 原创,本文整理了一部分的关键内容,希望对需要学习、面试 Linux 运维的同学有所帮助。
$1 入参,空值时默认赋值技巧
variable=${1:-"default value"} # 当未传参时,赋默认值。
echo $variable
$* 和 $@ 区别
- $* 和 $@ 都表示传递给函数或脚本的所有参数,不被双引号(" ")包含
时,都以"$1" "$2" … "$n" 的形式输出所有参数。 - 但是当它们被双引号(" ")包含时,"$*" 会将所有的参数作为一个整体,
以"$1 $2 … $n"的形式输出所有参数;" - $@" 会将各个参数分开,以"$1" "$2" … "$n" 的形式输出所有参数。
for var in "$*";do
echo "$var"
done
for var in "$@";do
echo "$var"
done
区别在于,分别输出 1 2 3 和
1
2
3
几个 Shell 里的内置变量
$# 入参个数
$$ PID
$0 文件名
$? 上个命令执行结果
知识扩展:
《awk 教程》 https://coolshell.cn/articles/9070.html
《sed 教程》 https://coolshell.cn/articles/9104.html
《Shell script 基础问答》 https://www.moewah.com/archives/1694.html
Shell 四则运算方法
let C=A+B
let A++ #let 时 变量前不需要再加$
echo $(($A+$B))
echo $[$A+$B]
expr $A + $B
echo $A+$B | bc #使用 bc 可以做比较复杂的运算
函数定义与调用
# 定义
function myfun(){ #function 可省略
echo $1 echo "-----"
echo $2 echo "-----"
echo $3
}
# 调用
myfun "tom" "lucy" "jack"
条件判断
if [[ biaodashi ]]
判断文件是否存在 -f 存在真
判断目录是否存在 -d 存在真
判断字符串是否空 -z 字符串长度为 0 真
判断文件大小 -s 文件大小非 0 真
if [[ 条件 1 -a 条件 2 ]] # -a 表示与, -o 表示或
字符串与序列、随机数
name=John && echo 'My name is $name' 输出'My name is $name ,因为单引号里面的$只会当作字面值
字符串截取 a=123456789 ; echo ${a:0:3} 输出 123
echo 替换字符
var=tom_tom_lucy_jack_tomomttom
echo ${var/tom/mary} mary_tom_lucy_jack_tomomttom
echo ${var//tom/mary} mary_mary_lucy_jack_maryomtmary
序列
for i in {1..100} #1,2,3,4...100 这种方法好些,还可以输出间
隔系列
for i in {1..100..7} #1,8,15....99
for i in `seq 100 ` #1,2,3,4...100
for i in `seq 3 5 ` #3,4,5
随机数
echo $RANDOM
head /dev/urandom |md5sum|cksum|cut -c 1-9
使用 cksum
从管道灌进去一些随机文本可生成一些随机数字,需要使用 cut
切 割
字符串截取
a=123A123B456456C
echo ${a##*3} 从最左边找到 3 删除,贪婪 B456456C
echo ${a%%6*} 从最右边找到 6 删除,贪婪 123A123B45
echo ${a%?} 删除最右边一个字符 123A123B456456
提取最后一位
str="12345678"
i=$((${#str}-1)) #字符串长度减去 1,故 i=7
echo ${b:$i:1} #输出 8
日期
date +%F" "%R # 2018-10-14 00:09
删除空行
cat 1.txt|tr -s "\n"
sed -i '/^$/d' 1.txt
grep -v "^$" 1.txt
Shell script 语法在线检查与建议: https://www.shellcheck.net
考察 if 、head 、tail、sed: https://leetcode.com/problems/tenth-line
考察 ls 按时间、体积大小排序
ls -lt 最新在前
ls -ltr 最旧在前
ls -lS 最大在前
ls -lSr 最小在前
考察 comm 的使用
命令详解 http://man.linuxde.net/comm
前提是,文件要排序过。结果的第一列是仅仅在文件 1 出现的,第二列是仅仅在文件 2 出现的,第三列是共同出现的。-1 表示不显示第一列,-2 表示不显示第二列,-3 表示不显示第三列。
考察 grep 与正则、子模式
https://leetcode.com/problems/valid-phone-numbers
https://www.interviewbit.com/problems/valid-phone-number/
答案 :
grep -P "(\d{3}-|\(\d{3}\)\s{1})\d{3}-\d{4}" input
另外, grep -c
表示匹配次数,-P
表示使用正则 -E
其实是扩展模式不是正则
考察容错处理、if、大于小于、正则、排序、去重统计、awk
https://leetcode.com/problems/word-frequency
grep
使用 -o
输出、使用-P
表示正则、使用 awk
交换列
答案:
grep -o -P "\w{1,}" words.txt |sort |uniq -c|sort -nr|awk '{print $2,$1}'
考察 awk 编程
给定一个文本文件,内有 M 行 N 列数字,求数字求和
题:给定 id 姓名 工资文本,计算工资和
1 tom 2500
2 mary 3200
3 jack 4700
4 who 6900
5 lee 2600
答案:
awk 'BEGIN{sum=0}{sum+=$3}END{print sum}' 1.txt
Ps:注意 BEGIN
、END
的位置;注意 awk
里定义的变量不用$
号
考察 awk 编程、单引号里如何传递变量
https://www.interviewbit.com/problems/lines-in-a-given-range/
awk 'NR>='$L' && NR<='$R' {print $0}' input
Ps:在单引号字句里,使用'$A'
以传递变量 A
使用 awk
一行检测磁盘分区> 90%
df -Ph | awk ' NR != 1 && $5 >= 90 ' # 为什么单引号里加{}不行呢
考察 tr 替换、删除的使用、去除空格
https://www.interviewbit.com/problems/remove-punctuations/
cat input|tr -d -c "a-zA-Z0-9[:space:]"
注意 tr
里什么代表字母、什么代表数字集
检测主机是否存活的脚本(三次 ping 都失败)
Ps:有个陷阱,ubuntu 16 里使用 sh 执行时,function xxx() 会报错。因此命令行用 bash 执行脚本
#!/bin/bash
function check_ip() { # 定义ping探测函数,function 可省略
fail_count=0
ip=$1
for (( i=0;i<3;i++ ));do # 使用 (( 表达式 )) 来做 for 循环 ping 3 次
if ping -W 1 -c 1 "$ip" > /dev/null 2>&1;then # 使用-W 1 避免很久超时,单位只能整数秒有效
break # 如果ping成功了,则直接结束探测,节省时间
fi
(( fail_count=fail_count+1 )) #若ping失败,则失败数累加,这里使用(( 表达式 )) 比 let 更加高效
done
if [[ $fail_count -eq 3 ]];then
echo "$ip is failed"
else
echo "$ip is ok"
fi
}
iplist="192.168.1.100 192.168.1.101" # 给出IP列表,当然也可以自行改造成文本方案
for ip in $iplist;do
check_ip $ip
done
批量并发检测存活主机
#!/bin/bash
for ip in 119.29.192.{1..255}; #批量 IP 的技巧 ;仅仅在 bash 生效,sh 不行
do
(
ping -c3 -W1 $ip >/dev/null ;
if [ $? -eq 0 ];then
echo "$ip alive"
fi
) & #使用( )& 挂在后台 并发,更快
done
wait #等待所有子进程结束
检测站点 URL 是否存活:考察 curl 的使用、文件入参、待优化多次检测
#!/bin/bash
# this script read urls from url.txt,then check whether the site isavailable
INPUT_FILE="url.txt"
while read u;do
curl -s --connect-timeout 3 -o /dev/null $u #注意几个参数的使用 --silent 可以用-s
if [ $? -ne 0 ];then
echo "$u"" failed."
else
echo "$u"" success"
fi
done < $INPUT_FILE
让进程在后台可靠运行的方法(意思是不会随着子 shell 的退出而挂)
nohup command > /dev/null 2>&1 &
screen -S <label> # 用快捷键 CTRL -a d 来暂时断开当前会话
command & #注意,这样标准输出可能会在屏幕里翻滚
十个常用脚本
1 获取随机字符串或数字
获取随机8位字符串:
方法1:
# echo $RANDOM |md5sum |cut -c 1-8
471b94f2
方法2:
# openssl rand -base64 4
vg3BEg==
方法3:
# cat /proc/sys/kernel/random/uuid |cut -c 1-8
ed9e032c
获取随机8位数字:
方法1:
# echo $RANDOM |cksum |cut -c 1-8
23648321
方法2:
# openssl rand -base64 4 |cksum |cut -c 1-8
38571131
方法3:
# date +%N |cut -c 1-8
69024815
cksum:打印CRC效验和统计字节
2 定义一个颜色输出字符串函数
方法1:
function echo_color() {
if [ $1 == "green" ]; then
echo -e "\033[32;40m$2\033[0m"
elif [ $1 == "red" ]; then
echo -e "\033[31;40m$2\033[0m"
fi
}
方法2:
function echo_color() {
case $1 in
green)
echo -e "\033[32;40m$2\033[0m"
;;
red)
echo -e "\033[31;40m$2\033[0m"
;;
*)
echo "Example: echo_color red string"
esac
}
使用方法:echo_color green "test"
function关键字定义一个函数,可加或不加。
3 批量创建用户
#!/bin/bash
DATE=$(date +%F_%T)
USER_FILE=user.txt
echo_color(){
if [ $1 == "green" ]; then
echo -e "\033[32;40m$2\033[0m"
elif [ $1 == "red" ]; then
echo -e "\033[31;40m$2\033[0m"
fi
}
# 如果用户文件存在并且大小大于0就备份
if [ -s $USER_FILE ]; then
mv $USER_FILE ${USER_FILE}-${DATE}.bak
echo_color green "$USER_FILE exist, rename ${USER_FILE}-${DATE}.bak"
fi
echo -e "User\tPassword" >> $USER_FILE
echo "----------------" >> $USER_FILE
for USER in user{1..10}; do
if ! id $USER &>/dev/null; then
PASS=$(echo $RANDOM |md5sum |cut -c 1-8)
useradd $USER
echo $PASS |passwd --stdin $USER &>/dev/null
echo -e "$USER\t$PASS" >> $USER_FILE
echo "$USER User create successful."
else
echo_color red "$USER User already exists!"
fi
done
4 检查软件包是否安装
#!/bin/bash
if rpm -q sysstat &>/dev/null; then
echo "sysstat is already installed."
else
echo "sysstat is not installed!"
fi
5 检查服务状态
#!/bin/bash
PORT_C=$(ss -anu |grep -c 123)
PS_C=$(ps -ef |grep ntpd |grep -vc grep)
if [ $PORT_C -eq 0 -o $PS_C -eq 0 ]; then
echo "内容" | mail -s "主题" [email protected]
fi
6 检查主机存活状态
方法1:将错误IP放到数组里面判断是否ping失败三次
#!/bin/bash
IP_LIST="192.168.18.1 192.168.1.1 192.168.18.2"
for IP in $IP_LIST; do
NUM=1
while [ $NUM -le 3 ]; do
if ping -c 1 $IP > /dev/null; then
echo "$IP Ping is successful."
break
else
# echo "$IP Ping is failure $NUM"
FAIL_COUNT[$NUM]=$IP
let NUM++
fi
done
if [ ${#FAIL_COUNT[*]} -eq 3 ];then
echo "${FAIL_COUNT[1]} Ping is failure!"
unset FAIL_COUNT[*]
fi
done
方法2:将错误次数放到FAIL_COUNT变量里面判断是否ping失败三次
#!/bin/bash
IP_LIST="192.168.18.1 192.168.1.1 192.168.18.2"
for IP in $IP_LIST; do
FAIL_COUNT=0
for ((i=1;i<=3;i++)); do
if ping -c 1 $IP >/dev/null; then
echo "$IP Ping is successful."
break
else
# echo "$IP Ping is failure $i"
let FAIL_COUNT++
fi
done
if [ $FAIL_COUNT -eq 3 ]; then
echo "$IP Ping is failure!"
fi
done
方法3:利用for循环将ping通就跳出循环继续,如果不跳出就会走到打印ping失败
#!/bin/bash
ping_success_status() {
if ping -c 1 $IP >/dev/null; then
echo "$IP Ping is successful."
continue
fi
}
IP_LIST="192.168.18.1 192.168.1.1 192.168.18.2"
for IP in $IP_LIST; do
ping_success_status
ping_success_status
ping_success_status
echo "$IP Ping is failure!"
done
7 监控CPU、内存和硬盘利用率
1)CPU
借助vmstat工具来分析CPU统计信息。
#!/bin/bash
DATE=$(date +%F" "%H:%M)
IP=$(ifconfig eth0 |awk -F '[ :]+' '/inet addr/{print $4}') # 只支持CentOS6
MAIL="[email protected]"
if ! which vmstat &>/dev/null; then
echo "vmstat command no found, Please install procps package."
exit 1
fi
US=$(vmstat |awk 'NR==3{print $13}')
SY=$(vmstat |awk 'NR==3{print $14}')
IDLE=$(vmstat |awk 'NR==3{print $15}')
WAIT=$(vmstat |awk 'NR==3{print $16}')
USE=$(($US+$SY))
if [ $USE -ge 50 ]; then
echo "
Date: $DATE
Host: $IP
Problem: CPU utilization $USE
" | mail -s "CPU Monitor" $MAIL
fi
2)内存
#!/bin/bash
DATE=$(date +%F" "%H:%M)
IP=$(ifconfig eth0 |awk -F '[ :]+' '/inet addr/{print $4}')
MAIL="[email protected]"
TOTAL=$(free -m |awk '/Mem/{print $2}')
USE=$(free -m |awk '/Mem/{print $3-$6-$7}')
FREE=$(($TOTAL-$USE))
# 内存小于1G发送报警邮件
if [ $FREE -lt 1024 ]; then
echo "
Date: $DATE
Host: $IP
Problem: Total=$TOTAL,Use=$USE,Free=$FREE
" | mail -s "Memory Monitor" $MAIL
fi
3)硬盘
#!/bin/bash
DATE=$(date +%F" "%H:%M)
IP=$(ifconfig eth0 |awk -F '[ :]+' '/inet addr/{print $4}')
MAIL="[email protected]"
TOTAL=$(fdisk -l |awk -F'[: ]+' 'BEGIN{OFS="="}/^Disk \/dev/{printf "%s=%sG,",$2,$3}')
PART_USE=$(df -h |awk 'BEGIN{OFS="="}/^\/dev/{print $1,int($5),$6}')
for i in $PART_USE; do
PART=$(echo $i |cut -d"=" -f1)
USE=$(echo $i |cut -d"=" -f2)
MOUNT=$(echo $i |cut -d"=" -f3)
if [ $USE -gt 80 ]; then
echo "
Date: $DATE
Host: $IP
Total: $TOTAL
Problem: $PART=$USE($MOUNT)
" | mail -s "Disk Monitor" $MAIL
fi
done
>
8 批量主机磁盘利用率监控
前提监控端和被监控端SSH免交互登录或者密钥登录。
写一个配置文件保存被监控主机SSH连接信息,文件内容格式:IP User Port
#!/bin/bash
HOST_INFO=host.info
for IP in $(awk '/^[^#]/{print $1}' $HOST_INFO); do
USER=$(awk -v ip=$IP 'ip==$1{print $2}' $HOST_INFO)
PORT=$(awk -v ip=$IP 'ip==$1{print $3}' $HOST_INFO)
TMP_FILE=/tmp/disk.tmp
ssh -p $PORT $USER@$IP 'df -h' > $TMP_FILE
USE_RATE_LIST=$(awk 'BEGIN{OFS="="}/^\/dev/{print $1,int($5)}' $TMP_FILE)
for USE_RATE in $USE_RATE_LIST; do
PART_NAME=${USE_RATE%=*}
USE_RATE=${USE_RATE#*=}
if [ $USE_RATE -ge 80 ]; then
echo "Warning: $PART_NAME Partition usage $USE_RATE%!"
fi
done
done
9 检查网站可用性
1)检查URL可用性
方法1:
check_url() {
HTTP_CODE=$(curl -o /dev/null --connect-timeout 3 -s -w "%{http_code}" $1)
if [ $HTTP_CODE -ne 200 ]; then
echo "Warning: $1 Access failure!"
fi
}
方法2:
check_url() {
if ! wget -T 10 --tries=1 --spider $1 >/dev/null 2>&1; then
#-T超时时间,--tries尝试1次,--spider爬虫模式
echo "Warning: $1 Access failure!"
fi
}
使用方法:check_url www.baidu.com
2)判断三次URL可用性
思路与上面检查主机存活状态一样。
方法1:利用循环技巧,如果成功就跳出当前循环,否则执行到最后一行
#!/bin/bash
check_url() {
HTTP_CODE=$(curl -o /dev/null --connect-timeout 3 -s -w "%{http_code}" $1)
if [ $HTTP_CODE -eq 200 ]; then
continue
fi
}
URL_LIST="www.baidu.com www.agasgf.com"
for URL in $URL_LIST; do
check_url $URL
check_url $URL
check_url $URL
echo "Warning: $URL Access failure!"
done
方法2:错误次数保存到变量
#!/bin/bash
URL_LIST="www.baidu.com www.agasgf.com"
for URL in $URL_LIST; do
FAIL_COUNT=0
for ((i=1;i<=3;i++)); do
HTTP_CODE=$(curl -o /dev/null --connect-timeout 3 -s -w "%{http_code}" $URL)
if [ $HTTP_CODE -ne 200 ]; then
let FAIL_COUNT++
else
break
fi
done
if [ $FAIL_COUNT -eq 3 ]; then
echo "Warning: $URL Access failure!"
fi
done
方法3:错误次数保存到数组
#!/bin/bash
URL_LIST="www.baidu.com www.agasgf.com"
for URL in $URL_LIST; do
NUM=1
while [ $NUM -le 3 ]; do
HTTP_CODE=$(curl -o /dev/null --connect-timeout 3 -s -w "%{http_code}" $URL)
if [ $HTTP_CODE -ne 200 ]; then
FAIL_COUNT[$NUM]=$IP #创建数组,以$NUM下标,$IP元素
let NUM++
else
break
fi
done
if [ ${#FAIL_COUNT[*]} -eq 3 ]; then
echo "Warning: $URL Access failure!"
unset FAIL_COUNT[*] #清空数组
fi
done
10 检查MySQL主从同步状态
#!/bin/bash
USER=bak
PASSWD=123456
IO_SQL_STATUS=$(mysql -u$USER -p$PASSWD -e 'show slave status\G' |awk -F: '/Slave_.*_Running/{gsub(": ",":");print $0}') #gsub去除冒号后面的空格
for i in $IO_SQL_STATUS; do
THREAD_STATUS_NAME=${i%:*}
THREAD_STATUS=${i#*:}
if [ "$THREAD_STATUS" != "Yes" ]; then
echo "Error: MySQL Master-Slave $THREAD_STATUS_NAME status is $THREAD_STATUS!"
fi
done
本章写的Shell脚本例子都比较实用,在面试题中也经常出现,希望大家参考着多动手写写,不要复制粘贴就拿来跑,这样是学不会的!
注意事项
1)开头加解释器:#!/bin/bash
2)语法缩进,使用四个空格;多加注释说明。
3)命名建议规则:变量名大写、局部变量小写,函数名小写,名字体现出实际作用。
4)默认变量是全局的,在函数中变量local指定为局部变量,避免污染其他作用域。
5)有两个命令能帮助我调试脚本:set -e 遇到执行非0时退出脚本,set-x 打印执行过程。
6)写脚本一定先测试再到生产上。