shell脚本概述(8)

一:shell

1.1:什么是shell

Shell 是一个用 C 语言编写的应用程序。

它类似于DOS下的command和后来的cmd.exe。它接收用户命令,然后调用相应的应用程序。

1.2:shell脚本

Shell 脚本(shell script),是一种为 shell 编写的脚本程序。

Shell脚本与Windows/Dos下的批处理相似,也就是将各类命令预先放入到一个文件中,方便一次性执行的一个程序文件。

主要是方便管理员进行设置或者管理用的。但是它比Windows下的批处理更强大。

1.3:shell脚本语法

  1. 开头:#!/bin/sh或#!/bin/bash
    符号#!是一个约定的标记,它告诉系统这个脚本需要什么解释器来执行
  2. 注释:
    #开头的行表示注释
  3. 命令行的书写规则:
    • 一行一条命令.
    • 若一行多个命令,用分号(;)分割.
    • 长命令可以使用反斜线字符(\).

1.4:第一个shell脚本

#!/bin/bash
echo "hello word!"

1.5:运行shell脚本

方法一:作为可执行程序
将上面的代码保存为 test.sh,并 cd 到相应目录:

chmod +x ./test.sh  #使脚本具有执行权限
./test.sh  #执行脚本

方法二:作为解释器参数
这种运行方式是,直接运行解释器,其参数就是 shell 脚本的文件名,如:

/bin/sh test.sh
或
sh test.sh

这种方式运行的脚本,不需要在第一行指定解释器信息,写了也没用。

二:变量

变量就是变化的量

2.1:系统变量

使用env命令可以查看操作系统的变量

查看某个具体的变量

[root@localhost ~]# echo $HOME
/root
[root@localhost ~]# echo $SHELL
/bin/bash

常用系统的变量:

系统定义的变量 意义
BASH Bash Shell 名称
HOME 用户家目录
OSTYPE 操作系统类型
PATH 可执行文件搜索路径
PWD 当前工作目录
USER 当前登录用户
RANDOM 生成随机数

设置新的环境变量:

用vim在/etc/profile文件中添加我们想要的环境变量
export 新环境变量名=内容

例:export BW=”wg007”
生效:source /etc/profile

2.2:自定义变量

name=“huazai007”

2.2.1:变量名的命名规则

  1. 命名只能使用英文字母,数字和下划线,首个字符不能以数字开头。
  2. 中间不能有空格,可以使用下划线(_)。
  3. 不能使用标点符号。
  4. 不能使用bash里的关键字(可用help命令查看保留关键字[echo,dir,pwd,read…])。

2.3:只读变量

使用 readonly 命令可以将变量定义为只读变量,只读变量的值不能被改变。

下面的例子尝试更改只读变量,结果报错:

#!/bin/bash
myname="huazai"
readonly myname
myUrl="huazai007"

运行脚本,结果如下:

/bin/sh: NAME: This variable is read only.

2.3:删除变量

使用 unset 命令可以删除变量。语法:

unset variable_name

变量被删除后不能再次使用。unset 命令不能删除只读变量。

2.4:特殊变量

特殊变量 说明
$0 Shell本身的文件名
$1~$n 添加到Shell的各参数值。$1是第1参数、$2是第2参数…。
$$ 脚本运行的当前进程ID号
$! 后台运行的最后一个进程的ID号
$* 以一个单字符串显示所有向脚本传递的参数。如"$*“用「”」括起来的情况、以"$1 $2 … $n"的形式输出所有参数。
$@ 与$*相同,但是使用时加引号,并在引号中返回每个参数.如"$@“用「”」括起来的情况、以"$1" “$2” … “$n” 的形式输出所有参数。
$# 传递到脚本的参数个数
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。

示例:

#!/bin/bash
echo "shell脚本的文件名为:$0"
echo "第一个参数的值为: $1"
echo "第二个参数的值为: $2"
echo "当前进程号为: $$"
echo "传递到脚本的参数个数为: $#"
echo "显示所有的传参:$@"
echo "显示所有的传参:$*"

执行结果:

[root@localhost ~]# sh t1.sh huazai 007
shell脚本的文件名为:t1.sh
第一个参数的值为: huazai
第二个参数的值为: 007
当前进程号为: 12793
传递到脚本的参数个数为: 2
显示所有的传参:huazai 007
显示所有的传参:huazai 007

$* 与 $@ 区别:

  • 相同点:都是引用所有参数。
  • 不同点:只有在双引号中体现出来。假设在脚本运行时写了三个参数 1、2、3,,则 " * " 等价于 “1 2 3”(传递了一个参数),而 “@” 等价于 “1” “2” “3”(传递了三个参数)。

示例:

#!/bin/bash
echo "-- \$* 演示 ---"
for i in "$*"
do
    echo $i
done

echo "-- \$@ 演示 ---"
for i in "$@"
do
    echo $i
done

执行结果:

[root@localhost ~]# sh t2.sh 1 2 3
-- $* 演示 ---
1 2 3
-- $@ 演示 ---
1
2
3

2.5:变量调用

在变量名前面加一个$符号

echo $name

2.6:read命令

read命令从键盘读取变量的值。
通常用在shell脚本中与用户进行交互的场合。

该命令可以一次读取多个变量的值,变量和输入的值都需要使用空格隔开。

在read命令后面,如果没有指定变量名,读取的数据将被自动赋值给特定的变量REPLY.【reply:答复】

read -p “提示信息” 变量名

例子:

read –p “请输入你的用户名” username

三:shell字符串

字符串是shell编程中最常用最有用的数据类型.

字符串可以用单引号,也可以用双引号,也可以不用引号。

3.1:单引号

str='this is a string'

单引号字符串的限制:

  • 单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的。
  • 单引号字串中不能出现单独一个的单引号,必须成对出现。

3.2:双引号

name="huazai007"
str="hello,$name\n"
echo -e $str

输出结果为:

hello,huazai007

双引号的优点:

  • 双引号里可以有变量
  • 双引号里可以出现转义字符(\n,\t)

3.3:获取字符串长度

[root@localhost ~]# str='huazai007'
[root@localhost ~]# echo ${#str}
9

四:shell数组

Shell 数组用括号来表示,元素用"空格"符号分割开,数组元素的下标由0开始。

4.1 定义一个数组:

语法格式如下:

array_name=(value1 value2 ... valuen)

方法一:

my_array=(A B C D)

方法二:
使用下标来定义数组:

my_array[0]=A
my_array[1]=B
my_array[2]=C

4.2 读取数组:

读取数组元素值的一般格式是:

${array_name[index]}

示例:

[root@localhost ~]# cat t3.sh 
#!/bin/bash
my_array=(A B C D)

echo "第一个元素为: ${my_array[0]}"
echo "第二个元素为: ${my_array[1]}"
echo "第三个元素为: ${my_array[2]}"
echo "第四个元素为: ${my_array[3]}"

执行结果:

[root@localhost ~]# sh t3.sh 
第一个元素为: A
第二个元素为: B
第三个元素为: C
第四个元素为: D

4.3 获取数组中的所有元素:

使用@ 或 * 可以获取数组中的所有元素,例如:

#!/bin/bash
my_array=(A B C D)
echo "数组的元素为: ${my_array[*]}"
echo "数组的元素为: ${my_array[@]}"

执行结果:

$ chmod +x test.sh 
$ ./test.sh
数组的元素为: A B C D
数组的元素为: A B C D

4.4 获取数组的长度:

获取数组长度的方法与获取字符串长度的方法相同,例如:

my_array[0]=A
my_array[1]=B
my_array[2]=C
my_array[3]=D

echo "数组元素个数为: ${#my_array[*]}"
echo "数组元素个数为: ${#my_array[@]}"

执行脚本,输出结果如下所示:

$ chmod +x test.sh 
$ ./test.sh
数组元素个数为: 4
数组元素个数为: 4

五:shell运算符

原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用。

注意:使用expr计算时,表达式和运算符之间要有空格

[root@localhost ~]# expr 1 + 2
3

[root@localhost ~]# echo $[1+3]
4
[root@localhost ~]# echo $((1+3))
4

5.1:算术运算符

常用的算术运算符:
假定变量 a 为 10,变量 b 为 20:

运算符 说明 举例
+ 加法 expr $a + $b 结果为 30。
- 减法 expr $a - $b 结果为 -10。
* 乘法 expr $a \* $b 结果为 200。
/ 除法 expr $b / $a 结果为 2。
% 取余 expr $b % $a 结果为 0。

5.2:关系运算符

关系运算符只支持数字,不支持字符串。

下表列出了常用的关系运算符,假定变量 a 为 10,变量 b 为 20:

运算符 说明 举例
-eq 检测两个数是否相等,相等返回 true。 [ $a -eq $b ] 返回 false。
-ne 检测两个数是否不相等,不相等返回 true。 [ $a -ne $b ] 返回 true。
-gt 检测左边的数是否大于右边的,如果是,则返回 true。 [ $a -gt $b ] 返回 false。
-lt 检测左边的数是否小于右边的,如果是,则返回 true。 [ $a -lt $b ] 返回 true。
-ge 检测左边的数是否大于等于右边的,如果是,则返回 true。 [ $a -ge $b ] 返回 false。
-le 检测左边的数是否小于等于右边的,如果是,则返回 true。 [ $a -le $b ] 返回 true。

关系运算符实例如下:

#!/bin/bash
# author:菜鸟教程
# url:www.runoob.com

a=10
b=20

if [ $a -eq $b ]
then
   echo "$a -eq $b : a 等于 b"
else
   echo "$a -eq $b: a 不等于 b"
fi
if [ $a -ne $b ]
then
   echo "$a -ne $b: a 不等于 b"
else
   echo "$a -ne $b : a 等于 b"
fi
if [ $a -gt $b ]
then
   echo "$a -gt $b: a 大于 b"
else
   echo "$a -gt $b: a 不大于 b"
fi
if [ $a -lt $b ]
then
   echo "$a -lt $b: a 小于 b"
else
   echo "$a -lt $b: a 不小于 b"
fi
if [ $a -ge $b ]
then
   echo "$a -ge $b: a 大于或等于 b"
else
   echo "$a -ge $b: a 小于 b"
fi
if [ $a -le $b ]
then
   echo "$a -le $b: a 小于或等于 b"
else
   echo "$a -le $b: a 大于 b"
fi

执行脚本,输出结果如下所示:

10 -eq 20: a 不等于 b
10 -ne 20: a 不等于 b
10 -gt 20: a 不大于 b
10 -lt 20: a 小于 b
10 -ge 20: a 小于 b
10 -le 20: a 小于或等于 b

5.2:布尔运算符

下表列出了常用的布尔运算符,假定变量 a 为 10,变量 b 为 20:

运算符 说明 举例
-a 与运算,两个表达式都为 true 才返回 true。 [ $a -lt 20 -a $b -gt 100 ] 返回 false。
-o 或运算,有一个表达式为 true 则返回 true。 [ $a -lt 20 -o $b -gt 100 ] 返回 true。
! 非运算,表达式为 true 则返回 false,否则返回 true。 [ ! false ] 返回 true。

实例如下:

a=10
b=20

if [ $a -lt 100 -a $b -gt 15 ]
then
   echo "$a 小于 100 且 $b 大于 15 : 返回 true"
else
   echo "$a 小于 100 且 $b 大于 15 : 返回 false"
fi
if [ $a -lt 100 -o $b -gt 100 ]
then
   echo "$a 小于 100 或 $b 大于 100 : 返回 true"
else
   echo "$a 小于 100 或 $b 大于 100 : 返回 false"
fi
if [ ! $a -gt $b ]; then
	echo "不大于"
else
	echo "小于或等于"
fi

5.3:逻辑运算符

以下介绍 Shell 的逻辑运算符,假定变量 a 为 10,变量 b 为 20:

运算符 说明 举例
&& 逻辑的 AND [[ $a -lt 100 && $b -gt 100 ]] 返回 false
|| 逻辑的 OR [[ $a -lt 100

逻辑运算符实例如下:

a=10
b=20

if [[ $a -lt 100 && $b -gt 100 ]]
then
   echo "返回 true"
else
   echo "返回 false"
fi

if [[ $a -lt 100 || $b -gt 100 ]]
then
   echo "返回 true"
else
   echo "返回 false"
fi

shell中[]和[[]]的区别:

[[]]支持通配符,[]不支持。

实例如下:

#!/bin/bash
for ip in `cat dropip`
do
        if [[ $ip =~ "192.168" ]] || [[ $ip =~ "127" ]]; then
                echo "pass"
        else
                echo "$ip"
                iptables -I INPUT -p tcp -s $ip -j DROP
        fi
done

5.4:字符串运算符

下表列出了常用的字符串运算符,假定变量 a 为 “abc”,变量 b 为 “efg”:

运算符 说明 举例
= 检测两个字符串是否相等,相等返回 true。 [ $a = $b ] 返回 false。
!= 检测两个字符串是否相等,不相等返回 true。 [ $a != $b ] 返回 true。
-z 检测字符串长度是否为0,为0返回 true。 [ -z $a ] 返回 false。
-n 检测字符串长度是否不为 0,不为 0 返回 true。 [ -n “$a” ] 返回 true。

字符串运算符实例如下:

#!/bin/bash
a="abc"
b="efg"

if [ $a = $b ]
then
   echo "$a = $b : a 等于 b"
else
   echo "$a = $b: a 不等于 b"
fi
if [ $a != $b ]
then
   echo "$a != $b : a 不等于 b"
else
   echo "$a != $b: a 等于 b"
fi
if [ -z $a ]
then
   echo "-z $a : 字符串长度为 0"
else
   echo "-z $a : 字符串长度不为 0"
fi
if [ -n "$a" ]
then
   echo "-n $a : 字符串长度不为 0"
else
   echo "-n $a : 字符串长度为 0"
fi

5.5:文件测试运算符

文件测试运算符用于检测 Unix 文件的各种属性。
做这个实验记得切换到普通用户!

属性检测描述如下:

操作符 说明 举例
-b fiile 检测文件是否是块设备文件,如果是,则返回 true。 [ -b $file ] 返回 false。
-c file 检测文件是否是字符设备文件,如果是,则返回 true。 [ -c $f
-d file 检测文件是否是目录,如果是,则返回 true。 [ -d $file ] 返回 false。
-f file 检测文件是否是普通文件(既不是目录,也不是设备文件),如果是,则返回 true。 [ -f $file ] 返回 true。
-r file 检测文件是否可读,如果是,则返回 true。 [ -r $file ] 返回 true。
-w file 检测文件是否可写,如果是,则返回 true。 [ -w $file ] 返回 true。
-x file 检测文件是否可执行,如果是,则返回 true。 [ -x $file ] 返回 true。
-s file 检测文件是否为空(文件大小是否大于0),不为空返回 true。 [ -s $file ] 返回 true。
-e file 检测文件(包括目录)是否存在,如果是,则返回 true。 [ -e $file ] 返回 true。

下面的代码,将检测该文件的各种属性:

#!/bin/bash

file="/shell/test.sh"
if [ -r $file ]
then
   echo "文件可读"
else
   echo "文件不可读"
fi
if [ -w $file ]
then
   echo "文件可写"
else
   echo "文件不可写"
fi
if [ -x $file ]
then
   echo "文件可执行"
else
   echo "文件不可执行"
fi
if [ -f $file ]
then
   echo "文件为普通文件"
else
   echo "文件为特殊文件"
fi
if [ -d $file ]
then
   echo "文件是个目录"
else
   echo "文件不是个目录"
fi
if [ -s $file ]
then
   echo "文件不为空"
else
   echo "文件为空"
fi
if [ -e $file ]
then
   echo "文件存在"
else
   echo "文件不存在"
fi

六:流程控制

6.1:if 语句

6.1.1:单分支if语句格式

if  [ 表达式 ]; then
		操作
fi

在这里插入图片描述if单分支实例:

在这里插入图片描述

6.1.2:双分支if语句格式

if [ 表达式 ]; then
	操作1
else
	操作2	
fi

在这里插入图片描述if双分支实例一:

在这里插入图片描述

#!/bin/bash
#判断当前用户是否为root
#如果是,则输出为“hello root”
#否则,切换为root用户
user=$USER
if [ $user = "root" ]; then
	echo "hello root"
else
	echo "开始切换root用户"
	su - root
fi

if双分支实例二:

#!/bin/bash
#编写监控httpd服务的脚本,判断httpd服务是否开启
#如未开启,则输出“httpd服务未开启,正在启动...”,并开启服务
#否则输出“httpd服务已经开启”
pgrep httpd >/dev/null
if [ $? -gt 0 ]; then
        echo "httpd服务未开启,正在启动..."
        systemctl start httpd
else
        echo "httpd服务已经开启"
fi

if双分支实例三:

#!/bin/bash
#同居当前登录到系统中的用户数量,判断是否超过3个
#若超过3个,则显示实际数量并给出警告信息
#否则列出用户名,tty号,时间日期,主机地址
num=`who|wc -l`
#num=$(who|wc -l)
if [ $num -gt 3 ]; then
        echo "当前登录用户数为$num,登录用户过多,请及时处理"
else
        who
fi

if双分支实例四:

#!/bin/bash
#将/var/log下的所有log文件拷贝到/logbak/当前日期/ 为名的目录下
#如果成功则输出“拷贝成功”
#如果失败则输出“拷贝失败”
#如果/logbak目录不存在则创建

if [ ! -d /logbak ]; then
        mkdir /logbak
fi

dirname=`date +%Y%m%d`
mkdir /logbak/$dirname
cp /var/log/*.log /logbak/$dirname

if [ $? -eq 0 ]; then
        echo "拷贝成功"
else
        echo "拷贝失败"
fi

if双分支实例五:

#!/bin/bash
#提示用户输入用户名和密码(tom+123)
#如果输入的用户名不是tom 或密码不是123,则提示“用户名或密码错误”
#否则登录成功

read -p "请输入用户名和秘密:<<" user pwd
if [ $user != "tom" -o $pwd != "123" ]; then
	echo "用户名或密码错误"
else
	echo "登录成功"
fi

6.1.3:多分支if语句格式

if [ 表达式 ]; then
	操作1
elif [ 表达式 ];then
	操作2
else
	操作3
fi

在这里插入图片描述if多分支语句案例:

两个数比大小

	#!/bin/bash
	prive=88
	read -p "请输入一个1-100的随机数:" num
    if [ $num -gt $prive ]; then
        echo "猜大了"
    elif [ $num -lt $prive ]; then
        echo "猜小了"
    else
        echo "两个数相等"
    fi

6.2:case 语句

6.2.1:case 语句格式

case 表达式 in
	值1|值2)     
	操作 ;;      
	值3|值4) 
	操作 ;;
	值5|值6) 
	操作 ;;
	*)
	操作;;
esac

case语句的语法规范:

  1. 表达式expr按顺序匹配每个模式,一旦匹配成功,则执行该模式后面的命令块,然后退出case语句
  2. 如果没有找到匹配的模式,则执行默认值“)”后的命令块, “)”可以没有
  3. 模式中可以包含通配符或“|”,如果多个模式对应同一个命令块,则使用“|”将各个模式分开
  4. 每个模式必须以右圆括号”)”结束
  5. 每个命令块必须以双分号(;;)结束,双分号可独占一行,也可放在最后一个命令的后面
  6. 以esac结尾,表示case语句的结束

在这里插入图片描述
case语句案例:
case语句为多选择语句。

可以用case语句匹配一个值与一个模式,如果匹配成功,执行相匹配的命令。

掌握case语句案例一:编写db服务的启动脚本
在这里插入图片描述case语句案例二:编写对数据服务的常规操作(增删改查)

read -p "请输入您想要执行的操作:<<" cmd
case $cmd in
        create)
        mysql -e "create table wg007.s1(id int,name varchar(20))"
        ;;
        insert)
        mysql -e "insert into wg007.s1 values(1,'hehe')"
        ;;
        delete)
        mysql -e "delete from wg007.s1 where id=1"
        ;;
        update)
        mysql -e "update wg007.s1 set name='haha' where id=1;"
        ;;
        select)
        mysql -e "select * from wg007.s1;"
        ;;
        *)
        echo "Usage: $0 {create|insert|delete|update|select}"
        ;;
esac

6.3:for循环语句

6.3.1:seq命令

用于产生从某个数到另外一个数之间的所有整数

seq 1 10
结果是1 2 3 4 5 6 7 8 9 10

6.3.2:for语句格式

格式1:

for  变量  in  列表
do
        操作
done

格式2:

for  ((  初始化表达式; 条件表达式; 更新循环变量表达式  ))
do
	循环语句
done

实例:
循环输出1到10
方法1:

for i  in   `seq  1   10`
do
        echo $i
done

方法2:

for (( i=1; i<=10; i++ ))
do
        echo $i
	done

创建user01、user02 …user10共10个用户

#!/bin/bash
for ((i=1;i<=10;i=i+1))
do
    echo "正在创建第$i个用户"
    useradd user$i
done

6.4:while循环语句

6.4.1:let命令

用于计算

[root@localhost ~]# count=0
#每执行一次加1
[root@localhost ~]# let count++ (或let count=count+1)
[root@localhost ~]# echo $count
1

6.4.2:while语句格式

注意:只有表达式为真,do和done之间的语句才会执行,表达式为假时,结束循环(即条件成立就一直执行循环)

while   表达式
do
 		 操作
done

循环输出1到10的数字

#!/bin/bash
num=1
while [ $num -le 10 ]
do
        echo $num
        let num++
done

死循环

whiledo
	echo “hehe”
done

6.5:until语句

注意:重复do和done之间的操作,直到表达式成立为止(即只要条件成立就停止执行循环)
表达式:

until  表达式
do
 	操作
done

例子:循环输出1到10的数字

#!/bin/bash
var=1
until [ $var -gt 10 ]
do
       echo $var
       myvar=$(( $var + 1 ))
done

6.6:break与continue语句

使用关键字“break”来跳出循环
使用关键字“continue”来不执行当前循环,然后跳到下一次循环继续执行

例子1:创建文件file1,file2,file4,file5

#!/bin/bash
for i in `seq 1 5`
do
	if [ $i –eq 3 ];then
		continue
	else
		touch file$i
	fi
done

例子2: 创建文件file1,file2

#!/bin/bash
for i in `seq 1 5`
do
	if [ $i -eq 3 ];then
		break
	else
		touch file$i
	fi
done

七:shell 函数

7.1: 什么是函数

将完成特定功能的代码封装,并命名,即函数,在使用时再进行调用

7.2:函数定义格式

说明:可以不带function 函数名()定义,也可以直接函数名()定义

function  函数名(   )
{
    
    
   函数体
}

函数名(   )
{
    
    
   函数体
}

7.3:函数调用方式为

函数名  参数列表   参数存放在$1-$9

编写一个求和函数,求两数之和

说明:给函数传递参数: 类比于脚本的位置参数

#!/bin/sh
add()
{
    
    
a=$1
b=$2
z=`expr $a + $b `
echo "The sum is $z"
}
add 8 9

执行结果:17

八:输入输出重定向

九:shell文件包含

猜你喜欢

转载自blog.csdn.net/zhangshaohuas/article/details/109009750