分类:
2012-08-24 12:22:05
原文地址:Shell脚本编程详解 作者:futter521
Shell命令行的书写规则
对Shell命令行基本功能的理解,有助于编写更好的Shell程序,在执行Shell命令时多个命令可以在一个命令行上运行,但此时要使用分号(;)分隔命令,例如:
[root@localhost root]# ls a* -l;free;df
长Shell命令行可以使用反斜线字符(\)在命令行上扩充,例如:
[root@localhost root]# echo “this is \
>longcommand”
This islong command
注意:
“>”符号是自动产生的,而不是输入的。
12-2 编写/修改权限及执行Shell程序的步骤①编写Shell程序
②执行Shell程序
Shell程序有很多类似C语言和其他程序设计语言的特征,但是又没有程序语言那样复杂。Shell程序是指放在一个文件中的一系列Linux命令 和实用程序。在执行的时候,通过Linux操作系统一个接一个地解释和执行每条命令。首先,来编写第一个Shell程序,从中学习Shell程序的编写、 修改权限、执行过程。
12-2-1 编辑Shell程序编辑一个内容如下的源程序,保存文件名为date,可将其存放在目录/bin下。
[root@localhost bin]#vi date
点击(此处)折叠或打开
注意:
#!/bin/sh 通知采用Bash解释。如果在echo语句中执行Shell命令date,则需要在date命令前加符号“&”,其中%B%d%A为输入格式控制符。
12-2-2 建立可执行程序编辑完该文件之后不能立即执行该文件,需给文件设置可执行程序权限。使用如下命令。
[root@localhost bin]#chmod +x date
12-2-3 执行Shell程序执行Shell程序有下面三种方法:
方法一:
[root@localhost bin]#./ date
Mr.root,Todayis:
二月 06 星期二
Wishyou a lucky day !
方法二:
另一种执行date的方法就是把它作为一个参数传递给Shell命令:
[root@localhost bin]# Bash date
Mr.root,Todayis:
二月 06 星期二
Wishyou a lucky day !
方法三:
为了在任何目录都可以编译和执行Shell所编写的程序,即把/bin的这个目录添加到整个环境变量中。
具体操作如下:
[root@localhost root]#export PATH=/bin:$PATH
[root@localhost bin]# date
Mr.root,Todayis:
二月06 星期二
Wish you a luckyday !
实例 12-1:编写一个Shell程序mkf,此程序的功能是:显示root下的文件信息,然后建立一个kk的文件夹,在此文件夹下建立一个文件aa,修改此文件的权限为可执行。
分析:此Shell程序中需要依次执行下列命令为:
进入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
位置参数
内部参数
如同ls命令可以接受目录等作为它的参数一样,在Shell编程时同样可以使用参数。Shell程序中的参数分为位置参数和内部参数等。
12-3-1 位置参数由系统提供的参数称为位置参数。位置参数的值可以用$N得到,N是一个数字,如果为1,即$1。类似C语言中的数组,Linux会把输入的命令字符 串分段并给每段进行标号,标号从0开始。第0号为程序名字,从1开始就表示传递给程序的参数。如$0表示程序的名字,$1表示传递给程序的第一个参数,以 此类推。
12-3-2 内部参数上述过程中的$0是一个内部变量,它是必须的,而$1则可有可无,最常用的内部变量有$0、$#、$?、$*,它们的含义如下。
$0:命令含命令所在的路径。
$#:传递给程序的总的参数数目。
$?:Shell程序在Shell中退出的情况,正常退出返回0,反之为非0值。
$*:传递给程序的所有参数组成的字符串。//参数空格是一行,不换行
$@:传递给程序的所有参数组成的字符串。//参数,以空格换行
$$:脚本运行的当前进程ID号
$!:后台运行的最后一个进程的进程ID号
实例 12-2:编写一个Shell程序,用于描述Shell程序中的位置参数为:$0、$#、$?、$*,程序名为test1,代码如下:
[root@localhost bin]#vi test1
执行后的结果如下:
[root@localhost bin]# test1 this is a test program //传递5个参数
注意:命令不计算在参数内。
实例 12-3:利用内部变量和位置参数编写一个名为test2的简单删除程序,如删除的文件名为a,则在终端中输入的命令为:test a
分析:除命令外至少还有一个位置参数,即$#不能为0,删除不能为$1,程序设计过程如下。
(1) 用vi编辑程序
[root@localhost bin]#vi test2
(2) 设置权限
[root@localhost bin]#chmod +x test2
(3) 运行
[root@localhost bin]# test2 a (如果a文件在bin目录下存在)
File a isdeleted!
12-4 在Shell程序中的使用变量变量的赋值
变量的访问
变量的输入
12-4-1变量的赋值在Shell编程中,所有的变量名都由字符串组成,并且不需要对变量进行声明。要赋值给一个变量,其格式如下:
变量名=值
注意:等号(=)前后没有空格
例如:
表示把6赋值给变量x,字符串“How are you ”赋值给变量a。
12-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
12-4-3 键盘读入变量值在Shell程序设计中,变量的值可以作为字符串从键盘读入,其格式为:
例如:
[root@localhost bin]#read str
read为读入命令,它表示从键盘读入字符串到str。
实例 12-4:编写一个Shell程序test3,程序执行时从键盘读入一个目录名,然后显示这个目录下所有文件的信息。
分析: 存放目录的变量为DIRECTORY,其读入语句为:
read DIRECTORY
显示文件的信息命令为:ls –a
[root@localhost bin]#vi test3
(2)设置权限
[root@localhost bin]#chmod +x test3
(3)执行
[root@localhost bin]#./test3
注意:
输入路径时需“/”
实例 12-5:运行程序test4,从键盘读入x、y的值,然后做加法运算,最后输出结果。
(1)用vi编辑程序
[root@localhost bin]#vi test4
(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`中的符号“`”为键盘左上角的“`”键。//即反引号
12-5 表达式的比较①字符串操作符
②逻辑运算符
③用test比较的运算符
④数字比较符
⑤文件操作符
在Shell程序中,通常使用表达式比较来完成逻辑任务。表达式 所代表的操作符有字符操作符、数字操作符、逻辑操作符、以及文件操作符。其中文件操作符是一种Shell所独特的操作符。因为Shell里的变量都是字符 串,为了达到对文件进行操作的目的,于是才提供了文件操作符。
12-5-1 字符串比较作用:测试字符串是否相等、长度是否为零,字符串是否为NULL。
常用的字符串操作符如表12-1所示.。
表12-1 常用的字符串操作符
字符串操作符 |
含义及返回值 |
= |
比较两个字符串是否相同,相同则为“真” |
!= |
比较两个字符串是否不相同,不同则为“真” |
-n |
比较两个字符串长度是否大于零,若大于零则为“真” |
-z |
比较两个字符串长度是否等于零,若等于零则为“真” |
实例 12-6:从键盘输入两个字符串,判断这两个字符串是否相等,如相等输出。
(1)用vi编辑程序
[root@localhost bin]#vi test5
(2)设置权限
[root@localhost bin]#chmod +x test5
(3)执行
[root@localhost root]#./ test5
aaa
bbb
1
注意:
”[”后面和”]”前面及等号“=“的前后都应有一个空格;注意这里是程序的退出情况,如果ar1和ar2的字符串是不相等的非正常退出,输出结果为1。
实例 12-7: 比较字符串长度是否大于零
(1)用vi编辑程序
[root@localhost bin]#vi test6
(2)设置权限
[root@localhost bin]#chmod +x test6
(3)执行
[root@localhost bin]#./ test6
0
注意:
运行结果1表示ar的小于等于零,0表示ar的长度大于零。
12-5-2 数字比较在Bash Shell编程中的关系运算有别于其他编程语言,用表12-2中的运算符用test语句表示大小的比较。
表12-2 用test比较的运算符
运算符号 |
含 义 |
-eq |
相等 |
-ge |
大于等于 |
-le |
小于等于 |
-ne |
不等于 |
-gt |
大于 |
-lt |
小于 |
实例 12-8:比较两个数字是否相等
(1)用vi编辑程序
[root@localhost bin]#vi test7
(2)设置权限
[root@localhost bin]#chmod +x test7
(3)执行
[root@localhost bin]#./ test7
50 100
50!=100
[root@localhost bin]#./ test7
150 150
150= =150
12-5-3 逻辑操作在Shell程序设计中的逻辑运算符如表12-3所示。
12-3 Shell中的逻辑运算符
运算符号 |
含 义 |
! |
反:与一个逻辑值相反的逻辑值 |
-a |
与(and):两个逻辑值为“是”返回值为“是”,反之为“否” |
-o |
或(or): 两个逻辑值有一个为“是”,返回值就是“是” |
实例 12-9:分别给两个字符变量赋值,一个变量赋予一定的值,另一个变量为空,求两者的与、或操作。
(1)用vi编辑程序
[root@localhost bin]#vi test8
(2)设置权限
[root@localhost bin]#chmod +x test8
(3)执行
[root@localhost bin]#./ test8
1
0
12-5-4 文件操作文件测试操作表达式通常是为了测试文件的信息,一般由脚本来决定文件是否应该备份、复制或删除。由于test关于文件的操作符有很多,在表12-4中只列举一些常用的操作符。
表12-4 文件测试操作符
运算符号 |
含 义 |
-d |
对象存在且为目录返回值为“是” |
-f |
对象存在且为文件返回值为“是” |
-L |
对象存在且为符号连接返回值为“是” |
-r |
对象存在且可读则返回值为“是” |
-s |
对象存在且长度非零则返回值为“是” |
-w |
对象存在且且可写则返回值为“是” |
-x |
对象存在且且可执行则返回值为“是” |
实例 12-10:判断zb目录是否存在于/root下。
(1)用vi编辑程序
[root@localhost bin]#vi test9
(2)设置权限
[root@localhost bin]#chmod +x test9
(3)执行
[root@localhost bint]#./ test9
(4)在/root添加zb目录
[root@localhost bin]#mkdir zb
(5)执行
[root@localhost bin]#./test9
0
注意:
运行结果是返回参数“$?”,结果1表示判断的目录不存在,0表示判断的目录不存在。
实例 12-11:编写一个Shell程序test10,输入一个字符串,如果是目录,则显示目录下的信息,如为文件显示文件的内容。
(1)用vi编辑程序
[root@localhost bin]#vi test10
[root@localhost bin]#chmod +x test10
(3)执行
[root@localhost bin]#./ test10
12-6 循环结构语句Shell的循环语句
Shell常见的循环语句有for循环、while循环语句和until循环。
12-6-1 for循环语法:
注意:
变量要在循环内部用来指列表当中的对象。
列表是在for循环的内部要操作的对象,可以是字符也串可以是文件,如果是文件则为文件名。
实例 12-12:在列表中的值:a,b,c,e,I,2,4,6,8,用循环的方式把字符与数字分成两行输出。
(1)用gedit编辑脚本程序test11
[root@localhost bin]#gedit test11
(2)设置权限
[root@localhost bin]#chmod +x test11
(3)执行
[root@localhost bin]#./ test11
a,b,c,e,i
2,4,6,8
注意:
在循环列表中的空格可表示换行。
实例 12-13:删除垃圾箱中的所有文件。
分析:在本机中,垃圾箱的位置是在$HOME/.Trash中,因而是在删除$HOME/.Trash列表当中的所有文件,程序脚本如下。
(1)用gedit编辑脚本程序test12
[root@localhost bin]#gedit test12
(2)设置权限
[root@localhost bin]#chmod +x test12
(3)执行
[root@localhost bin]#./ test12
/root/.Trash/abc~has been deleted!
/root/.Trash/abc1has been deleted!
实例 12-14:求从1~100的和。
(1)用gedit编辑脚本程序test13
[root@localhost bin]#gedit test13
(2)设置权限
[root@localhost bin]#chmod +x test13
(3)执行
[root@localhost bin]#./ test13
The result is5050
注意:
for语句中的双括号不能省,最后的分号可有可无,表达式total=`expr $total + $j`的加号两边的空格不能省,否则会成为字符串的连接。
12-6-2while循环语法:
只要表达式为真,do和done之间的操作就一直会进行。
实例 12-15:用while循环求1~100的和。
(1)用gedit编辑脚本程序test14
[root@localhost bin]#gedit test13
(2)设置权限
[root@localhost bin]#chmod +x test14
(3)执行
[root@localhost bin]#./ test14
The result is5050
12-6-3until循环语法:
重复do和done之间的操作直到表达式成立为止。
实例 12-16:用until循环求1~100的和。
(1)用gedit编辑脚本程序test15
[root@localhost bin]#gedit test15
(2)设置权限
[root@localhost bin]#chmod +x test15
(3)执行
[root@localhost bin]#./ test15
The result is5050
12-7 条件结构语句Shell的条件结构语句
Shell程序中的条件语句主要有if语句与case语句。
12-7-1 if语句语法:
Linux里的if的结束标志是将if反过来写成fi;而elif其实是else if的缩写。其中,elif理论上可以有无限多个。
实例 12-17:用for循环求1~100的和。
(1)用gedit编辑脚本程序test16
[root@localhost bin]#gedit test16
(2)设置权限
[root@localhost bin]#chmod +x test16
(3)执行
[root@localhost bin]#./ test16
13579
12-7-2 case语句语法:
case的作用就是当字符串与某个值相同是就执行那个值后面的操作。如果同一个操作对于多个值,则使用“|”将各个值分开。在case的每一个操作的最后面都有两个“;;”分号是必需的。
实例 12-18:Linux是一个多用户操作系统,编写一程序根据不同的用户登录输出不同的反馈结果。
(1)用vi编辑脚本程序test17
[root@localhost bin]#gedit test17
(2)设置权限
[root@localhost bin]#chmod +x test17
(3)执行
[root@localhost bin]#./ test17
You are root
Welcome!
12-8 在Shell脚本中使用函数Shell的函数
Shell程序也支持函数。函数能完成一特定的功能,可以重复调用这个函数。
函数格式如下:
函数调用方式为:
函数名 参数列表
实例 12-19:编写一函数add求两个数的和,这两个数用位置参数传入,最后输出结果。
(1)编辑代码
[root@localhost bin]#gedit test18
(2)设置权限
[root@localhost bin]#chmod +x test18
(3)执行
[root@localhost bin]#./ test18 10 20
The sum is 30
注意:
函数定义完成后必须同时写出函数的调用,然后对此文件进行权限设定,在执行此文件。
12-9 在Shell脚本中调用其他脚本Shell脚本的调用
在Shell脚本的执行过程中,Shell脚本支持调用另一个Shell脚本,调用的格式为:
程序名
实例 12-20:在Shell脚本test19中调用test20。
(1)调用test20
(2)设置权限
[root@localhost bin]#chmod +x test19
[root@localhost bin]#chmod +x test20
(3)执行
[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)在对函数命名时最好能使用有含义且能容易理解的名字,即使函数名能够比较准确地表达函数所完成的任务。同时建议对于较大的程序要建立函数名和变量命名对照表。
12-10本章小结本章讲解了Linux下Shell脚本的定义和相关Shell脚本编写的基础,这些基础知识是学习Shell脚本编程的关键。接着讲解了Shell 脚本的执行方式和Shell脚本的常见流程控制,为Shell脚本的编程做了准备。
课程实训实训内容:编写一个Shell程序,呈现一个菜单,有0-5共6个命令选项,1为挂载U盘,2为卸载U盘,3为显示U盘的信息,4把硬盘中的文件拷贝到U盘,5把U盘中的文件拷贝到硬盘中,选0为退出。
程序分析:把此程序分成题目中要求的6大功能模块,另外加一个菜单显示及选择的主模板。
(1) 编辑代码
[root@localhost bin]#vi test19
(2)修改权限
[root@localhost bin]#chmod +x test19
(3)程序执行结果
[root@localhost bin]#./ test19
项目实践这段时间学习Linux下的Shell 编程,感觉Shell 编程和C语言很相似。王工程师今天来看陈飞,顺便问一下陈飞的学习情况。陈飞就和他说了自己对Shell编程的看法。王工程师听了后,笑着说,“一样不一 样,你编个程序不久明白了吗。”“那编什么程序呢”。陈飞问道。“就俄罗斯方块吧”,王工程师说。“俄罗斯方块大家都会玩,而且你可以在网上找到用C语言 编写的程序,你用Shell编程实现,和C语言版的对比一下,不就明白了它们之间的不同了吗”。王工程师走了,留下了陷入沉思的陈飞。他能完成吗?
课后习题1. 选择题
(1) 下列说法中正确的是( )。
A.安装软件包fctix-3.4.tar.bz2,要按顺序使用./configure;make;make install;tar命令
B.挂载U盘,mount /dev/sda /mnt/u -o iocharset=gb2312
C.显示变量PS1的值用命令 echo PS1
D.用命令./abc与sh abc执行Shell脚本abc,所得的结果并不相同
(2) 一个Bash Shell脚本的第一行是什么(A)。
A.#!/bin/Bash B.#/bin/Bash C.#/bin/csh D./bin/Bash
(3) 在Shell脚本中,用来读取文件内各个域的内容并将其赋值给Shell变量的命令是(D )。
A. fold B. join C. tr D. read
(4) 下列变量名中有效的Shell变量名是(C)。
A. -2-time B. _2$3 C.trust_no_1 D. 2004file
(5) 下列对Shell变量FRUIT操作,正确的是(C )。
A. 为变量赋值:$FRUIT=apple B. 显示变量的值:fruit=apple
C. 显示变量的值:echo $FRUIT D. 判断变量是否有值:[ -f “$FRUIT” ]
(6) 在Fedora 12系统中,下列关于Shell脚本程序说法不正确的是(B )。
A. Shell脚本程序以文本的形式存储
B. Shell脚本程序在运行前需要进行编译
C. Shell脚本程序由解释程序解释执行
D. Shell 脚本程序主要用于系统管理和文件操作,它能够方便自如地处理大量重复性的系统工作
(7) 在Shell编程中关于$2的描述正确的是(C )。
A. 程序后携带了两个位置参数
B. 宏替换
C. 程序后面携带的第二个位置参数
D 携带位置参数的个数
E 用$2引用第二个位置参数
(8) 在Fedora 12系统中,“run.sh”是Shell执行脚本,在执行./run.sh file1 file2 file3的命令的过程中,变量$1的值为(B)。
A. run.sh B. file1 C. file2 D. file3
2. 填空题
(1) 在Shell编程时,使用方括号表示测试条件的规则是 "["后空格,"]"前空格 。
(2) 编写的Shell程序运行前必须赋予该脚本文件 可执行 权限。
3. 简答题
(1) 用Shell编程,判断一文件是不是字符设备文件,如果是将其拷贝到 /dev 目录下。
(2) 在根目录下有四个文件m1.txt,m2.txt,m3.txt,m4.txt,用Shell编程,实现自动创建m1,m2,m3,m4四个目录,并将m1.txt,m2.txt,m3.txt,m4.txt四个文件分别拷贝到各自相应的目录下。
(3) 某系统管理员需每天做一定的重复工作,请按照下列要求,编制一个解决方案:
①在下午4 :50删除/abc目录下的全部子目录和全部文件;
②从早8:00~下午6:00每小时读取/xyz目录下x1文件中每行第一个域的全部数据加入到/backup目录下的bak01.txt文件内;
③每逢星期一下午5:50将/data目录下的所有目录和文件归档并压缩为文件:backup.tar.gz;
④在下午5:55将IDE接口的CD-ROM卸载(假设:CD-ROM的设备名为hdc);
⑤在早晨8:00前开机后启动。
(4) 请用Shell编程来实现:当输入不同的选择时,执行不同的操作,如:输入start 开始启动应用程序myfiles,输入stop时,关闭myfiles,输入status时,查看myfiles进程,否则执行*)显示“EXIT!”并退出程序。
(5) 编写一个Shell程序,此程序的功能是:显示root下的文件信息,然后建立一个abc的文件夹,在此文件夹下建立一个文件k.c,修改此文件的权限为可执行。
(6) 编写一个Shell程序,挂载U盘,在U盘中根目录下所有.c文件拷贝到当前目录,然后卸载U盘。
(7) 编写一个Shell程序,程序执行时从键盘读入一个文件名,然后创建这个文件。
(8) 编写一个Shell程序,键盘输入两个字符串,比较两个字符串是否相等。
(9) 编写三个Shell程序,分别用for、while、与until求从2+4+…+100的和。
(10) 编写一个Shell程序,键盘输入两个数及+、-、*、与/中的任一运算符,计算这两个数的运算结果。
(11) 编写两个Shell程序kk及aa,在kk中输入两个数,调用aa计算计算这两个数之间奇数的和。
(12) 编写Shell程序,可以挂载U盘,也可挂载Windows硬盘的分区,并可对文件进行操作。
(13) 编写4个函数分别进行算术运算+、-、*、/,并编写一个菜单,实现运算命令。