shell脚本
1.shell命令书写规则
对Shell命令行基本功能的理解有助于编写更好的Shell程序,在执行Shell命令时多个命令可以在一个命令行上运行,但此时要使用分号(;)分隔命令,例如:
[root@localhost root]# ls a* -l;free;df
长Shell命令行可以使用反斜线字符(\)在命令行上扩充,例如:
[root@localhost root]# echo “this is \
>long command”
This is long command
注意:“>”符号是自动产生的,而不是输入的。
2.编写/修改权限及执行Shell程序的步骤
Shell程序是指放在一个文件中的一系列Linux命令和实用程序。在执行的时候,通过Linux操作系统一个接一个地解释和执行每条命令。
2.1编写shell程序
编辑一个内容如下的源程序,保存文件名为date,可将其存放在目录/bin下。
#! /bin/sh
echo “Mr.$USER,Today is:”
echo &date “+%B%d%A”
echo “Wish you a lucky day !”
2.2建立可执行程序
[root@localhost bin]#chmod +x date
2.3实例
编写一个Shell程序mkf,此程序的功能是:显示root下的文件信息,然后建立一个kk的文件夹,在此文件夹下建立一个文件aa,修改此文件的权限为可执行。 进入root目录:cd /root 显示root目录下的文件信息:ls –l 新建文件夹kk: mkdir kk 进入root/kk目录:cd kk 新建一个文件aa: vi aa #编辑完成后需手工保存 修改aa文件的权限为可执行:chmod +x aa 回到root目录:cd /root 因此该Shell程序只是以上命令的顺序集合,假定程序名为mkf
[root@localhost root]#vi mkf
cd /root
ls –l
mkdir kk
cd kk
vi aa
chmod +x aa
cd /root
3 在shell程序中使用参数
位置参数 与 内部参数
3.1 位置参数
由系统提供的参数称为位置参数。位置参数的值可以用$N得到,N是一个数字,如果为1,即$1。类似C语言中的数组,Linux会把输入的命令字符串分段并给每段进行标号,标号从0开始。第0号为程序名字,从1开始就表示传递给程序的参数。如$0表示程序的名字,$1表示传递给程序的第一个参数,以此类推。
3.2 内部参数
上述过程中的$0是一个内部变量,它是必须的,而$1则可有可无,最常用的内部变量 有$0、$#、$?、$*,它们的含义如下。
$0:命令含命令所在的路径。
$#:传递给程序的总的参数数目。
$?:Shell程序在Shell中退出的情况,正常退出返回0,反之为非0值。
$*:传递给程序的所有参数组成的字符串。
实例1:编写一个程序,用于描述Shell程序中的位置参数为:$0、$#、$?、$*,程序名为test1,代码如下:
[root@localhost bin]#vi test1
#! /bin/sh
echo “Program name is $0”;
echo “There are totally $# parameters passed to this program”;
echo “The last is $?”;
echo “The parameter are $*”;
执行后的结果如下:
[root@localhost bin]# test1 this is a test program //传递5个参数
Program name is /bin/test1 //给出程序的完整路径和名字
There are totally 5 parameters passed to this program //参数的总数
The last is 0 //程序执行效果
The parameters are this is a test program //返回由参数组成的字符串
实例2:利用内部变量和位置参数编写一个名为test2的简单删除程序,如删除的(当前目录下)文件名为a,则在终端中输入的命令为:test a
分析:除命令外至少还有一个位置参数,即$#(参数个数)不能为0,删除不能为$1,程序设计过程如下。
[root@localhost bin]#vi test2
#! /bin/sh
if test $# -eq 0
then
echo “Please specify a file!”
else
gzip $1 //现对文件进行压缩
mv $1.gz $HOME/dustbin //移动到回收站
echo “File $1 is deleted !”
fi
[root@localhost bin]#chmod +x test2
[root@localhost bin]# test2 a (如果a文件在bin目录下存在)
File a is deleted!
4 在Shell中的使用变量
变量的赋值、变量的访问、变量的输入
4.1 变量赋值
在Shell编程中,所有的变量名都由字符串组成,并且不需要对变量进行声明。要赋值给一个变量,其格式如下:
变量名=值
注意:
等号(=)前后没有空格
例如:
x=6
a=”How are you ”
表示把6赋值给变量x,字符串“How are you ”赋值给变量a。
4.2 访问变量
如果要访问变量值,可以在变量前面加一个美元符号“$”,例如:
[root@localhost bin]#a=”How are you ”
[root@localhost bin]#echo “He juest said:$a”
A is:hello world
一个变量给另一个变量赋值可以写成:
变量2=$变量1
例如:
x=$i
i++可以写成:
i=$i+1
4.3 键盘读入变量值
在Shell程序设计中,变量的值可以作为字符串从键盘读入,其格式为:
read 变量
例如:
[root@localhost bin]#read str
read为读入命令,它表示从键盘读入字符串到str。
实例1:编写一个Shell程序test3,程序执行时从键盘读入一个目录名,然后显示这个目录下所有文件的信息。
分析:存放目录的变量为DIRECTORY,其读入语句为:
read DIRECTORY
显示文件的信息命令为:
ls –a
代码
[root@localhost bin]#vi test3
#! /bin/sh
echo “please input name of directory”
read DIRECTORY
cd $DIRECTORY
ls –l
(2)设置权限
[root@localhost bin]#chmod +x test3
(3)执行
[root@localhost bin]#./test3
注意:
输入路径时需“/”
实例2: 运行程序test4,从键盘读入x、y的值,然后做加法运算,最后输出结果。
[root@localhost bin]#vi test4
#! /bin/sh
echo “please input x y”
read x,y
z=`expr $x+$y`
echo “The sum is $z”
(2)设置权限
[root@localhost bin]#chmod +x test4
(3)执行
[root@localhost bin]#./ test4
45 78
The sum is 123
注意:
表达式total=`expr $total +$num`及num=`expr $num +1`中的符号“`”为键盘左上角的“`”键。
5. 表达式的比较
字符串操作符 逻辑运算符 用test比较的运算符 数字比较符 文件操作符
5.1 字符串比较
作用:测试字符串是否相等、长度是否为零,字符串是否为NULL。 表12-1 常用的字符串操作符
字符串操作符 含义及返回值
= 比较两个字符串是否相同,相同则为“真”
!= 比较两个字符串是否不相同,不同则为“真”
-n 比较两个字符串长度是否大于零,若大于零则为“真”
-z 比较两个字符串长度是否等于零,若等于零则为“真”
实例1 :从键盘输入两个字符串,判断这两个字符串是否相等,如相等输出。
[root@localhost bin]#vi test5
#! /bin/Bash
read ar1
read ar2
[ “$ar1” = “$ar2” ]
echo $? #?保存前一个命令的返回码
结果:
aaa
bbb
1
注意:
”[”后面和”]”前面及等号“=“的前后都应有一个空格;注意这里是程序的退出情况,如果ar1和ar2的字符串是不想等的非正常退出,输出结果为1。
实例2: 比较字符串长度是否大于零
[root@localhost bin]#vi test6
#! /bin/Bash
read ar
[ -n “$ar” ]
echo $? //保存前一个命令的返回码
5.2 数字比较
在Bash Shell编程中的关系运算有别于其他编程语言,用表12-2中的运算符用test语句表示大小的比较。
表12-2 用test比较的运算符
运算符号 含 义
-eq 相等
-ge 大于等于
-le 小于等于
-ne 不等于
-gt 大于
-lt 小于
实例2. 比较两个数字是否相等
[root@localhost bin]#vi test7
#! /bin/Bash
read x,y
if test $x –eq $y
then
echo “$x=$y”
else
echo “$x!=$y”
fi
[root@localhost bin]#./ test7
50 100
50!=100
[root@localhost bin]#./ test7
150 150
150= =150
5.3逻辑操作
12-3 Shell中的逻辑运算符
运算符号 含 义
! 反:与一个逻辑值相反的逻辑值
-a 与(and):两个逻辑值为“是”返回值为“是”,反之为“否”
-o 或(or): 两个逻辑值有一个为“是”,返回值就是“是”
实例1: 分别给两个字符变量赋值,一个变量赋予一定的值,另一个变量为空,求两者的与、或操作。
[root@localhost bin]#vi test8
#! /bin/Bash
part1 =”1111”
part2 =” ” #part2为空
[ “$ part1” –a “$ part2”]
echo $? #保存前一个命令的返回码
[ “$ part1” –o “$ part2”]
echo $?
[root@localhost bin]#./ test8
1
0
5.4 文件操作
文件测试操作表达式通常是为了测试文件的信息,一般由脚本来决定文件是否应该备份、复制或删除。由于关于文件的操作符有很多,在表12-4中只列举一些常用的操作符。
表12-4 文件测试操作符
运算符号 含 义
-d 对象存在且为目录返回值为“是”
-f 对象存在且为文件返回值为“是”
-L 对象存在且为符号连接返回值为“是”
-r 对象存在且可读则返回值为“是”
-s 对象存在且长度非零则返回值为“是”
-w 对象存在且且可写则返回值为“是”
-x 对象存在且且可执行则返回值为“是”
实例: 判断zb目录是否存在于/root下。
[root@localhost bin]#vi test9
#! /bin/Bash
[ -d /root/zb ]
echo $? #保存前一个命令的返回码
注意:
运行结果是返回参数“$?”,结果1表示判断的目录不存在,0表示判断的目录不存在。
实例: 编写一个Shell程序test10,输入一个字符串,如果是目录,则显示目录下的信息,如为文件显示文件的内容。
[root@localhost bin]#vi test10
#! /bin/Bash
echo “Please enter the directory name or file name”
read DORF
if [ -d $DORF ]
then
ls $DORF
elif [ -f $DORF ]
then
cat $DORF
else
echo “input error! ”
fi
6 循环结构语句
Shell的循环语句、Shell常见的循环语句有for循环、while循环语句和until循环。
6.1 for循环
for 变量 in 列表
do
操作
done
注意:
变量要在循环内部用来指列表当中的对象。列表是在for循环的内部要操作的对象,可以是字符串也可以是文件,如果是文件则为文件名。
实例1: 在列表中的值:a,b,c,e,I,2,4,6,8用循环的方式把字符与数字分成两行输出。
#! /bin/Bash
for i in a,b,c,e,I 2,4,6,8
do
echo $i
done
root@localhost bin]#./ test11
a,b,c,e,i
2,4,6,8
实例2: 删除垃圾箱中的所有文件。
#! /bin/Bash
for i in $HOME/.Trash/*
do
rm $ i
echo “$ i has been deleted!”
done
实例3: 求从1~100的和。
#! /bin/Bash
total=0
for((j=1;j<=100;j++));
do
total=`expr $total + $j`
done
echo “The result is $total”
注意:
for语句中的双括号不能省,最后的分号可有可无,表达式total=`expr $total + $j`的加号两边的空格不能省,否则会成为字符串的连接。
6.2 while循环
语法:
while 表达式
do
操作
done
实例:用while循环求1~100的和。
[root@localhost bin]#gedit test13
total =0
num=0
while((num<=100));
do
total=’expor $total +$ num’
done
echo “The result is $total”
6.3 until循环
until 表达式
do
操作
done
实例: 用until循环求1~100的和。
total =0
num=0
until [$num –gt 100]
do
total=’expor $total +$ num’
num=’expr $num + 1’
done
echo “The result is $total”
6.7 if条件结构语句
if 表达式1 then
操作
elif表达式2 then
操作
elif表达式3 then
操作
……
else
操作
fi
实例: 用for循环求1~100的和。
for((j=0;j<=10;j++))
do
if(($j%2==1))
then
echo “$j”
fi
done
6.8 case语句
case 表达式 in
值1|值2)
操作;;
值3|值4)
操作;;
值5|值6)
操作;;
*)
操作;;
esac
8 shell中使用函数
函数名( )
{
函数体
}
函数调用方式为
函数名 参数列表
实例: 编写一函数add求两个数的和,这两个数用位置参数传入,最后输出结果。
#!/bin/sh
add()
{
a=$1
b=$2
z=’expr $a + $b’
echo “The sum is $z”
}
add $1 $2
[root@localhost bin]#./ test18 10 20
The sum is 30
注意:
函数定义完成后必须同时写出函数的调用,然后对此文件进行权限设定,在执行此文件。
9 shell中调用其他脚本
在Shell脚本的执行过程中,Shell脚本支持调用另一个Shell脚本,调用的格式为:
程序名
实例: 在Shell脚本test19中调用test20。
#test19脚本
#!/bin/sh
echo “The main name is $0”
./test20
echo “The first string is $1”
#test20脚本
#!/bin/sh
echo “How are you $USER?”
结果:
root@localhost bin]#./ test19 abc123
The main name is ./test19
How are you root?
the first string is abc123
注意:
1)在Linux编辑中命令区分大小写字符。
2)在Shell语句中加入必要的注释,以便以后查询和维护,注释以#开头。
3)对Shell变量进行数字运算时,使用乘法符号“*”时,要用转义字符“\”进行转义。
4)由于Shell对命令中多余的空格不进行任何处理,因此程序员可以利用这一特性调整程序缩进,达到增强程序可读性效果。
5)在对函数命名时最好能使用有含义且能容易理解的名字,即使函数名能够比较准确地表达函数所完成的任务。同时建议对于较大的程序要建立函数名和变量命名对照表。
题目
编写一个Shell程序,呈现一个菜单,有0-5共6个命令选项,1为挂载U盘,2为卸载U盘,3为显示U盘的信息,4把硬盘中的文件拷贝到U盘,5把U盘中的文件拷贝到硬盘中,选0为退出。
程序分析:把此程序分成题目中要求的6大功能模块,另外加一个菜单显示及选择的主模板。
(1) 编辑代码
[root@localhost bin]#vi test19
#!/bin/sh
#mountusb.sh
#退出程序函数
quit()
{
clear
echo “*******************************************************************”
echo “*** thank you to use,Good bye! ****”
exit 0
}
#加载U盘函数
mountusb()
{
clear
#在/mnt下创建usb目录
mkdir /mnt/usb
#查看U盘设备名称
/sbin/fdisk –l |grep /dev/sd
echo –e “Please Enter the device name of usb as shown above:\c”
read PARAMETER
mount /dev/$PARAMETER /mnt/usb
}
#卸载U盘函数
umountusb()
{
clear
ls -la /mnt/usb
}
#显示U盘信息函数
display()
{
clear
umount /mnt/usb
}
#拷贝硬盘文件到U盘函数
cpdisktousb()
{
clear
echo –e “Please Enter the filename to be Copide (under Current directory):\c”
read FILE
echo “Copying,please wait!...”
cp $FILE /mnt/usb
}
#拷贝U盘函数到硬盘文件
cpusbtodisk()
{
clear
echo -e “Please Enter the filename to be Copide in USB:\c”
read FILE
echo “Copying ,Please wait!...”
cp /mnt/usb/$FILE . #点(.)表示当前路径
}
clear
while true
do
echo “=====================================================================”
echo “*** LINUX USB MANAGE PROGRAM ***”
echo “ 1-MOUNT USB ”
echo “ 2-UNMOUNT USB ”
echo “ 3-DISPLAY USB INFORMATION ”
echo “ 4-COPY FILE IN DISK TO USB ”
echo “ 5-COPY FILE IN USB TO DISK ”
echo “ 0-EXIT ”
echo “=====================================================================”
echo –e “Please Enter a Choice(0-5):\c”
read CHOICE
case $CHOICE in
1) mountusb
2) unmountusb
3) display
4) cpdisktousb
5) cpusbtodisk
0) quit
*) echo “Invalid Choice!Corrent Choice is (0-5)”
sleep 4
clear;;
esac
done
(2)修改权限
[root@localhost bin]#chmod +x test19
(3)程序执行结果
[root@localhost bin]#./ test19