13、Linux 教程:shell脚本基本命令

shell脚本基本命令

输出命令echo

输出命令echo,基本模式就是echo [选项] [输出内容]输出内容如果包含空格,则必须将内容用双引号括起来。选项-e可以使输出语句支持反斜线转义。

 

加入退格后就不会显示退格符左边的一个字符。ascii码表中有对应的八进制和十六进制表示法,所以可以表示对应的字符。

显示环境变量的值:echo ${PATH}echo $PATH,如果一个变量没有被设定,那么就什么都不返回。

颜色输出如将abcd用红色打印:echo -e "\e[1;31m abcd \e[0m"其中\e[1的意思是开启颜色输出,而\e[0m是结束颜色输出,31m代表红色,abcd是输出内容,其他颜色如下:

 

第一个脚本与脚本执行方式

新建一个脚本hello.sh:

#!/bin/bash
#the first program

echo "hello world"
exit 0

其中第一行是声明,不是注释,不能省略,这是在指定使用哪个shell,如果没有这行有的程序可能无法执行。

第二行#开头的是注释,第四行是命令。

最后一行在设置回传值,在执行完该脚本后,执行echo $?就能查看这个值,可以通过这个自定义错误信息。

在脚本中有需要时要重新定义一下PATH环境变量,以便直接使用一些外部命令而不是写绝对路径:

PATH=/usr/lib64/qt-3.3/bin:/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/opt/module/jdk1.8.0_144/bin:/opt/module/hadoop-2.7.2/bin:/opt/module/hadoop-2.7.2/sbin:/root/bin
export PATH

执行shell脚本要先赋予其可rx权限:chomd 755 hello.sh然后再执行./hello.sh这里也可以用绝对路径执行。或者用bash命令执行(这种执行方法只要对脚本有r权限就可以执行):bash hello.sh,此时屏幕上就会打印hello world,sh hello.sh也一样。也可以将文件放入PATH指定的路径中,直接执行文件名就能运行了。(直接执行一串指令:sh -c "指令1;指令2"

上述几种脚本执行方式都属于直接执行,这些执行方式本质上会使用一个新的bash环境来运行指令,这个子shell不会继承父shell的普通变量,只能继承环境变量,而子shell中定义的变量也不会在父shell中查到,在脚本运行完毕后,子shell中所有的数据都会移除。

另外一种脚本执行方式是source指令,该指令后跟脚本路径就能执行,source执行脚本时不会开启子shell,所以各指令的运行结果都会在原shell中生效,脚本中定义了变量,运行完毕后依然可以查到,所以该命令常常用在修改配置文件后。

写脚本要养成一些良好的习惯,在开头记录一些关键的注释:功能、版本信息、作者及联络方式、历史记录。重要的代码也需要注释。

条件判断

判断文件类型:

 

基本判断格式有两种:test -e /root/install.log[ -e /root/install.log ]注意后者方括号内两边都必须有空格(这样处理主要是为了规避正则表达式)。这种判断语句直接执行不会显示正确还是错误,需要执行$?来查看该条命令执行结果。或者可以使用[ -e /root/install.log ] && echo "yes" || echo "no"根据打印结果来查看是否执行正确。

判断文件权限:

 

这种方式只能判断有没有这种权限,而不会区分具体是属主、属组或者other拥有这种权限,想要细分权限需要自定义脚本提取ll命令的打印结果。

两个文件比较:

 

两个整数之间进行比较:

 

字符串或变量的判断:

 

如果先执行name=sc再执行[ -z "$name" ] && echo "yes" || echo "no"则会输出no,因为此时变量name不为空。如果将$name替换为$不存在的变量,那么结果就为yes。

如果执行aa=abcbb=abc,然后再执行[ "$aa" == "$bb" ]结果就为真,[ "$aa" == abc ]结果也为真,但是[ "$aa" == "bb" ]结果为假。

在条件判断式方括号中有一些需要经常遵循的原则:

1、 每个组件之间都要用空格隔开(为了防止两边没有空格格式错误);

2、 变量都要用双引号括起来,常量用单引号或双引号括起来;

第二点原则其实和命令中的量用双引号括起来的原因是一样的,一旦出现空格就会发生一些错误,如执行:

[a b=="$c" ],其实不是a b整体和c比较,而是b在和c比较,这个命令会直接报错,应该这样执行:

["a b"=="$c" ]

(在bash的判断中使用一个等号和使用两个结构是一样的,但是推荐用两个等号)

多重条件判断:

 

if语句

单分支if条件语句:

if [ 条件判断式 ];then
程序
fi

if [ 条件判断式 ]
then
程序
fi

注意if中的条件判断式方括号两边还是要有空格。

在if语句中,方括号和方括号之间也可以加逻辑运算符&&和||,下面两行是等价的:

[ 判断1 -o 判断2 ]
[ 判断1 ] || [ 判断2 ]

双分支if条件语句:

if [ 条件判断式 ]
then
条件成立时,执行的程序
else
条件不成立时,执行的另一个程序
fi

多分支if条件语句:

if [ 条件判断式1 ] 
	then 
		当条件判断式1成立时,执行程序1 
elif [ 条件判断式2 ] 
	then 
		当条件判断式2成立时,执行程序2 
else 
	当所有条件都不成立时,最后执行此程序
fi

例1:判断分区使用率

#!/bin/bash
#统计根分区使用率
rate=$(df -h | grep "/dev/sda3" | awk '{print $5}' | cut -d "%" -f1)
#把根分区使用率作为变量值赋予变量rate
if [ $rate -ge 80 ]
	then
		echo "Warning! /dev/sda3 is full!!"
fi

这个脚本是基于查看分区情况命令df -h的,从输出内容中提取对应分区的对应列,然后去掉%,将值赋值给rate,然后用if语句判断rate的值是否大于80,如果大于就打印警告信息,实际应用中这里应该给管理员发送邮件。

例2:备份mysql数据库

#!/bin/bash
#备份mysql数据库。
ntpdate asia.pool.ntp.org &>/dev/null
#同步系统时间
date=$(date +%y%m%d)
#把当前系统时间按照“年月日”格式赋予变量date
size=$(du -sh /var/lib/mysql)
#统计mysql数据库的大小,并把大小赋予size变量
if [ -d /tmp/dbbak ]
	then
		echo "Date : $date!" > /tmp/dbbak/dbinfo.txt
		echo "Data size : $size" >> /tmp/dbbak/dbinfo.txt
		cd /tmp/dbbak
		tar -zcf mysql-lib-$date.tar.gz /var/lib/mysql dbinfo.txt &>/dev/null
		rm -rf /tmp/dbbak/dbinfo.txt
else
	mkdir /tmp/dbbak
	echo "Date : $date!" > /tmp/dbbak/dbinfo.txt
	echo "Data size : $size" >> /tmp/dbbak/dbinfo.txt
	cd /tmp/dbbak
	tar -zcf mysql-lib-$date.tar.gz /var/lib/mysql dbinfo.txt &>/dev/null
	rm -rf /tmp/dbbak/dbinfo.txt
fi

首先用if语句判断日志目录/tmp/dbbak是否存在,如果不存在就新建该目录。

将date和size变量都输出到dbinfo.txt文件中,然后将数据库/var/lib/mysql和dbinfo.txt都打包到mysql-lib-$date.tar.gz文件中,这个文件的命名有一部分调用了date的值,这是为了保持压缩文件文件名的唯一性,在打包过程中将命令执行过程中的打印内容全部输出到/dev/null中,相当于删除,&>表示无论语句执行正确与否都将打印内容重定向。打包完成后删掉dbinfo.txt文件。

如果想获取前两天的日期:

date2=$(date --date='2 days ago' +%Y%m%d)

文件名也可以单独拿出来拼接,这样可读性更强:

filename=${filename2}${date2}

例3:判断apache是否启动

#!/bin/bash
port=$(nmap -sT 192.168.1.156 | grep tcp | grep http | awk '{print $2}')
#使用nmap命令扫描服务器,并截取apache服务的状态,赋予变量port
if [ "$port" == "open" ]
	then
		echo “$(date) httpd is ok!” >> /tmp/autostart-acc.log
else
	/etc/rc.d/init.d/httpd start &>/dev/null
	echo "$(date) restart httpd !!" >> /tmp/autostart-err.log
fi

nmap命令是判断进程执行的重要命令,提取此命令的打印结果,得到占用状态:

 

然后根据状态进行if判断,如果已经启动就将已启动输出到日志,如果没有启动就开启该服务,然后将已启动命令输出到日志。

例4:判断用户输入的文件类型

#!/bin/bash
#判断用户输入的是什么文件
read -p "Please input a filename: " file
#接收键盘的输入,并赋予变量file
if [ -z "$file" ]
#判断file变量是否为空
	then
		echo "Error,please input a filename"
		exit 1
elif [ ! -e "$file" ]
#判断file的值是否存在
	then
		echo "Your input is not a file!"
		exit 2
elif [ -f "$file" ]
#判断file的值是否为普通文件
	then
		echo "$file is a regulare file!"
elif [ -d "$file" ]
#判断file的值是否为目录文件
	then
		echo "$file is a directory!"
else
		echo "$file is an other file!"
fi

先用read命令接受键盘输入,然后将输入赋值给file,用if判断file的类型,这里用exit来跳出多条件if语句,否则如果文件名为空的话还会继续执行下一个条件。

case语句

case语句和多分支if语句类似,不同之处在于case语句判断的条件关系只有一个。使用格式:

case $变量名 in
	"值1")
		如果变量的值等于值1,则执行程序1
		;;
	"值2")
		如果变量的值等于值2,则执行程序2
		;;
	*)
		如果变量的值都不是以上的值,则执行此程序
		;;
esac

function函数

基本格式:

function fname(){
	程序
}

fname就是函数名,也是调用时要执行的命令名。

注意在bash中调用函数一定要在定义函数之后,否则会报错。定义好函数后就可以在后续程序中直接调用fname命令,此时就相当于执行代码块。调用该命令时也可以跟后续的参数,第一个参数是$1、第二个参数是$2..,在function中$0代表函数名称,注意在函数内和函数外内建变量的值是不同的。

for循环

for格式1:

for 变量 in 值1 值2 值3
	do
		程序
	done

这种循环的循环次数和in后的值个数相等。in后面可以放:

1、 特殊符号,如$(seq1100)代表从1-100这中间的一百个值;

2、 多行结果,如cut命令执行的结果;

users=$(cut -d ':' -f1 /etc/passwd)
for username in ${users}
	do
		id ${username}
	done

for格式2:

for((初始值;循环控制条件;变量变化))
	do
		程序
	done

例1:批量解压缩脚本

#!/bin/bash

cd /lamp
ls *.tar.gz > ls.log
for i in $(cat ls.log)
	do
		tar -zxf $i &> /dev/null
	done
rm -rf /lamp/ls.log

将文件中的压缩包都重定向到日志文件中,然后把日志文件的内容放在in后,逐个执行解压,最后删除日志文件。

例2:for版1到100求和

#!/bin/bash
s=0
for((i=1;i<=100;i=i+1))
	do
		s=$(($s+$i))
	done
echo s

例3:批量添加用户

#!/bin/bash

read -p "input username :" -t 30 name
read -p "input the number of users: " -t 30 num
read -p "input the password of users: " -t 30 pass
if[ ! -z "$name" -a ! -z "$num" -a ! -z "$pass" ]
	then
		y=$(echo $num | sed 's/[0-9]//g')
			if[ -z "$y" ]
				then
					for((i=1;i<=$num;i=i+1))
						do
							useradd $name$i &> /dev/null
							echo $pass | passwd --stdin $name&i &> /dev/null
						done
			fi
fi

首先用户输入用户名name和要添加的用户数量num,还有用户密码pass,然后判断这三个变量是否为空,如果不为空那么就判断num是否为数字,这里用的方法是把num中的数字全部替换为空,然后再检查是否为空。最后开始循环,循环次数就是num,添加用户名时为了保持每个用户名不同设置名字时要加上循环次数,每个用户名为$name&i,设置密码也同理。

while和until循环

while循环,条件为假时退出循环。

while [ 条件判断式 ] 
	do 
		程序 
	done

until循环和while类似,不同之处在于条件为真时退出循环。

until [ 条件判断式 ]
	do
		程序
	done

例1:while版1到100求和

#!/bin/bash
#从1加到100
i=1
s=0
while [ $i -le 100 ]
#如果变量i的值小于等于100,则执行循环
do
s=$(( $s+$i ))
i=$(( $i+1 ))
done
echo "The sum is: $s"

脚本的debug

测试脚本是否有语法问题:sh -n 脚本路径,如果没有问题则不会显示任何信息。

列出脚本的执行过程:sh -x 脚本路径,输出的结果中加号开头的行是指令的执行过程,没有加号的代表打印内容。