191. Linux的shell脚本及用法

在linux中,Bash脚本是很基础的知识,虽然复杂的脚本是很烧脑,但是,当我们熟练的掌握了其中的用法与技巧,再多加练习,总有一天也会成为得心应手的脚本大神。脚本在生产中的作用,可以省下很多复杂的操作,减轻自己的工作压力。

条件选择、判断

条件选择if

(1)用法格式

1
2
3
4
5
6
7
8
9
if 判断条件 1 ; then
  #条件为真的分支代码
elif 判断条件 2 ; then
  #条件为真的分支代码
elif 判断条件 3 ; then
  #条件为真的分支代码
else
  #以上条件都为假的分支代码
fi

逐条件进行判断,第一次遇为“真”条件时,执行其分支,而后结束整个if。

(2)经典案例:

  1. 判断年纪
1
2
3
4
5
6
7
8
9
10
11
12
13
#!/bin/bash
read -p "Please input your age: " age
if [[ $age =~ [^0-9] ]] ;then
echo "please input a int"
exit 10
elif [ $age -ge 150 ];then
echo "your age is wrong"
exit 20
elif [ $age -gt 18 ];then
echo "good good work,day day up"
else
echo "good good study,day day up"
fi

分析:请输入年纪,先判断输入的是否含有除数字以外的字符,有,就报错;没有,继续判断是否小于150,是否大于18。

  1. 判断分数
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#!/bin/bash
read -p "Please input your score: " score
if [[ $score =~ [^0-9] ]] ;then
echo "please input a int"
exit 10
elif [ $score -gt 100 ];then
echo "Your score is wrong"
exit 20
elif [ $score -ge 85 ];then
echo "Your score is very good"
elif [ $score -ge 60 ];then
echo "Your score is soso"
else
echo "You are loser"
fi

分析:请输入成绩,先判断输入的是否含有除数字以外的字符,有,就报错;没有,继续判断是否大于100,是否大于85,是否大于60。

条件判断 case

(1)用法格式

1
2
3
4
5
6
7
8
9
10
11
case $name in;
PART1)
  cmd
  ;;
PART2)
  cmd
  ;;
*)
  cmd
  ;;
esac
1
2
3
4
5
注意:case 支持glob 风格的通配符:
  *: 任意长度任意字符
  ?: 任意单个字符
  [] :指定范围内的任意单个字符
  a|b: a 或b

(2)案例:

判断yes or no

1
2
3
4
5
6
7
8
9
10
11
12
13
#!/bin/bash
read -p "Please input yes or no: " anw
case $anw in
[yY][eE][sS]|[yY])
echo yes
;;
[nN][oO]|[nN])
echo no
;;
*)
echo false
;;
esac

分析:请输入yes or no,回答Y/y、yes各种大小写组合为yes;回答N/n、No各种大小写组合为no。

循环

for

(1)用法格式

1
2
3
4
#第一种方式
for name in 列表 ;do
   #循环体
done
1
2
3
4
#第二种
for (( exp1; exp2; exp3 )) ;do
   #cmd
done

exp1只执行一次,相当于在for里嵌了while

执行机制:依次将列表中的元素赋值给“变量名”; 每次赋值后即执行一次循环体; 直到列表中的元素耗尽,循环结束

列表的表示方法,可以glob 通配符,如{1..10} 、*.sh ;也可以变量引用,如: seq 1 $name

(2)案例

  1. 求出(1+2+…+n)的总和
1
2
3
4
5
6
7
8
9
10
11
12
13
sum=0
read -p "Please input a positive integer: " num
if [[ $num =~ [^0-9] ]] ;then
echo "input error"
elif [[ $num -eq 0 ]] ;then
echo "input error"
else
for i in `seq 1 $num` ;do
sum=$[$sum+$i]
done
echo $sum
fi
unset zhi

分析:sum初始值为0,请输入一个数,先判断输入的是否含有除数字以外的字符,有,就报错;没有判断是否为0,不为0进入for循环,i的范围为1~输入的数,每次的循环为sum=sum+i,循环结束,最后输出sum的值。

  1. 求出(1+2+…+100)的总和
1
2
3
4
for (( i=1,num=0;i<=100;i++ ));do
[ $[i%2] -eq 1 ] && let sum+=i
done
echo sum=$sum

分析:i=1,num=0;当i<=100,进入循环,若i÷2取余=1,则sum=sum+i,i=i+1。

while

(1)用法格式

1
2
3
while 循环控制条件 ;do
  #循环
done

循环控制条件;进入循环之前,先做一次判断;每一次循环之后会再次做判断;条件为“true” ,则执行一次循环;直到条件测试状态为“false” 终止循环

(2)特殊用法(遍历文件的每一行):

1
2
3
while read line; do控制变量初始化
  循环体
done < /PATH/FROM/SOMEFILE
1
2
3
cat /PATH/FROM/SOMEFILE | while read line; do
  循环体
done

依次读取/PATH/FROM/SOMEFILE文件中的每一行,且将行赋值给变量line

(3)案例:

  1. 100以内所有正奇数之和
1
2
3
4
5
6
7
8
9
10
11
sum=0
i=1
while [ $i -le 100 ] ;do
if [ $[$i%2] -ne 0 ];then
let sum+=i
let i++
else
let i++
fi
done
echo "sum is $sum"

分析:sum初始值为0,i的初始值为1;请输入一个数,先判断输入的是否含有除数字以外的字符,有,就报错;没有当i<100时,进入循环,判断 i÷2取余 是否不为0,不为0时为奇数,sum=sum+i,i+1,为0,i+1;循环结束,最后输出sum的值。

until 循环

(1)用法

1
2
3
unitl 循环条件 ;do
  循环
done

进入循环:循环条件为false ;退出循环:循环条件为true;刚好和while相反,所以不常用,用while就行。

(2)案例

监控xiaoming用户,登录就杀死

1
2
3
4
until pgrep -u xiaoming &> /dev/null ;do
sleep 0.5
done
pkill -9 -u xiaoming

分析:每隔0.5秒扫描,直到发现xiaoming用户登录,杀死这个进程,退出脚本,用于监控用户登录。

4、select 循环与菜单

(1)用法

1
2
3
4
select variable in list
do
  循环体命令
done
  1. select 循环主要用于创建菜单,按数字顺序排列的示菜单项将显示在标准错误上,并显示PS3 提示符,等待用户输入
  2. 用户输入菜单列表中的某个数字,执行相应的命令
  3. 用户输入被保存在内置变量 REPLY
  4. select 是个无限循环,因此要记住用 break 命令退出循环,或用 exit 按 命令终止脚本。也可以按 ctrl+c退出循环
  5. select 和 经常和 case 联合使用
  6. 与for循环类似,可以省略 in list, 此时使用位置参量

(2)案例:

生成菜单,并显示选中的价钱

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
PS3="Please choose the menu: "
select menu in mifan huimian jiaozi babaozhou quit
do
case $REPLY in
1|4)
echo "the price is 15"
;;
2|3)
echo "the price is 20"
;;
5)
break
;;
*)
echo "no the option"
esac
done

分析:PS3是select的提示符,自动生成菜单,选择5break退出循环。

循环里的一些用法

循环控制语句

(1)语法

continue [N]:提前结束第N层的本轮循环,而直接进入下一轮判断;最内层为第1层

break [N]:提前结束第N层循环,最内侧为第1层

1
2
3
4
5
6
7
while CONDTITON1; do
  CMD1
if CONDITION2; then
  continue / break
fi
  CMD2
done

(2)案例:

  1. 求(1+3+…+49+53+…+100)的和
1
2
3
4
5
6
7
#!/bin/bash
sum=0
for i in {1..100} ;do
[ $i -eq 51 ] && continue
[ $[$i%2] -eq 1 ] && { let sum+=i;let i++; }
done
echo sum=$sum

分析:做1+2+…+100的循环,当i=51时,跳过这次循环,但是继续整个循环,结果为:sum=2449

  1. 求(1+3+…+49)的和
1
2
3
4
5
6
7
#!/bin/bash
sum=0
for i in {1..100} ;do
[ $i -eq 51 ] && break
[ $[$i%2] -eq 1 ] && { let sum+=i;let i++; }
done
echo sum=$sum

分析:做1+2+…+100的循环,当i=51时,跳出整个循环,结果为:sum=625

循环控制shift命令

(1)作用

用于将参数列表list左移指定次数,最左端的那个参数就从列表中删除,其后边的参数继续进入循环

(2)案例:

  1. 创建指定的多个用户
1
2
3
4
5
6
7
8
9
10
#!/binbash
if [ $# -eq 0 ] ;then
echo "Please input a arg(eg:`basename $0` arg1)"
exit 1
else
while [ -n "$1" ];do
useradd $1 &> /dev/null
shift
done
fi

分析:如果没有输入参数(参数的总数为0),提示错误并退出;反之,进入循环;若第一个参数不为空字符,则创建以第一个参数为名的用户,并移除第一个参数,将紧跟的参数左移作为第一个参数,直到没有第一个参数,退出。

  1. 打印直角三角形的字符
1
2
3
4
5
6
#!/binbash
while (( $# > 0 ))
do
echo "$*"
shift
done

返回值结果

true 永远返回成功结果

: null command ,什么也不干,返回成功结果

false 永远返回错误结果

创建无限循环

1
2
3
while true ;do
  循环体
done

循环中可并行执行,使脚本运行更快

(1)用法

1
2
3
4
5
6
for name in 列表 ;do
{
  循环体
}&
done
wait

(2)实例:

搜寻自己指定ip(子网掩码为24的)的网段中,UP的ip地址

1
2
3
4
5
6
7
8
9
10
11
12

read -p "Please input network (eg:192.168.0.0): " net
echo $net |egrep -o "\<(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\>"
[ $? -eq 0 ] || ( echo "input error";exit 10 )
IP=`echo $net |egrep -o "^([0-9]{1,3}\.){3}"`
for i in {1..254};do
{
ping -c 1 -w 1 $IP$i &> /dev/null && \
echo "$IP$i is up"
}&
done
wait

分析:请输入一个IP地址例192.168.37.234,如果格式不是0.0.0.0 则报错退出;正确则进入循环,IP变量的值为192.168.37. i的范围为1-254,并行ping 192.168.37.1-154,ping通就输出此IP为UP。直到循环结束。

字符串

在我们熟知语言中都有自己的数据类型,那么shell是数据类型有哪些?shell属于弱类型语言,它不会强制你声明变量的数据类型。当给一个变量赋值时,根据赋值的内容自动确定数据类型。虽然不能直接定义数据类型,但是我们可以通过一些方法来判断变量的值是否符合某种类型的预期。常见的类型有:字符串、数值、数组、关联数组(map)、日期、布尔、空值。本文主要介绍字符串。

declare

在介绍类型前,先回顾下declare命令. 它可以指定变量的属性,通过-p参数显示每个变量的属性和值

格式如下:

1
declare [-aAfFgilrtux] [-p] [name[=value]] 

这里仅简单介绍与本文相关的属性:

属性 说明
-a 表示当前变量数据类型为数组.
-A 表示当前变量数据类型为关联数组.
-i 表示当前变量数据类型为整数.
-- 表示当前变量数据类型为字符串,默认

其他相关参数细节参见Shell变量定义与基本操作

定义

本节介绍3中定义字符串方式以及区别:双引号、单引号、无引号

双引号

1
2
3
# 通过declate -p命令查看变量属性
[root@ ~]# s1="a" ; declare -p s1
declare -- s1="a"

单引号

1
2
[root@ ~]# s2='b' ; declare -p s2
declare -- s2="b"

需要注意,单引号中的字符不会被转义,保持原样输出,参见条件测试操作符单引号和双引号区别

1
2
3
4
5
[root@ ~]# s11="a-$s1" ; echo $s11
a-a

[root@ ~]# s22='b-$s2' ; echo $s22
b-$s2

无引号

1
2
3
# 无引号,直接赋值
[root@ ~]# s3=c ; declare -p s3
declare -- s3="c"

基本操作

主要介绍字符串常见的操作:字符串长度、字符串截取、字符串匹配、字符串替换、字符串拼接、字符串去空格

字符串长度

格式:`$