Bourne Shell及Shell编程(2) h.交互式从键盘读入数据 使用read语句,其格式如下:
read var1 var2 ... varn
read将不作变量替换,但会删除多余的空格,直到遇到第一个换行符(回车), 并将输入值依次赋值给相应的变量。
例: $ read var1 var2 var3 Hello my friends $ echo $var1 $var2 $var3 Hello my friends $ echo $var1 Hello $ read var1 var2 var3 Hello my dear friends $ echo $var3 dear friends <-输入多余变量时,输入值余下的内容赋给最后一个变量 $ read var1 var2 var3 Hello friends $ echo $var3 <- var3为空 $
在shell script中可使用read语句进行交互操作:
... #echo -n message 输出结果后不换行 echo -n "Do you want to continue: Y or N" read ANSWER
if [ $ANSWER=N -o $ANSWER=n ] then exit fi
i. case结构:结构较elif-then结构更清楚
比较if-then语句:
if [ variable1 = value1 ] then command command elif [ variable1 = value2 ] then command command elif [ variable1 = value3 ] then command command fi
相应的case结构:
case value in pattern1) command command;; pattern2) command command;; ... patternn) command; esac
* case语句只执行第一个匹配模式
例:使用case语句建立一个菜单选择shell script
#Display a menu echo _ echo "1 Restore" echo "2 Backup" echo "3 Unload" echo
#Read and excute the user's selection echo -n "Enter Choice:" read CHOICE
case "$CHOICE" in 1) echo "Restore";; 2) echo "Backup";; 3) echo "Unload";; *) echo "Sorry $CHOICE is not a valid choice exit 1 esac
在上例中,*指默认匹配动作。此外,case模式中也可以使用逻辑操作,如下所示:
pattern1 | pattern2 ) command command ;;
这样可以将上面示例程序中允许用户输入数字或每一个大写字母。
case "$CHOICE" in 1|R) echo "Restore";; 2|B) echo "Backup";; 3|U) echo "Unload";; *) echo "Sorry $CHOICE is not a valid choice exit 1 esac
(5)循环控制 <1> while循环: 格式: while command do command command command ... done
例: 计算1到5的平方 #!/bin/sh # #Filename: square.sh int=1
while [ $int -le 5 ] do sq=`expr $int \* $int` echo $sq int=`expr $int + 1` done echo "Job completed"
$ sh square.sh 1 4 9 16 25 Job completed
<2> until循环结构: 格式: until command do command command .... command done
示例:使用until结构计算1-5的平方 #!/bin/sh
int=1
until [ $int -gt 5 ] do sq=`expr $int \* $int` echo $sq int=`expr $int + 1` done echo "Job completed"
<3> 使用shift对不定长的参数进行处理 在以上的示例中我们总是假设命令行参数唯一或其个数固定,或者使用$*将整个命令 行参数传递给shell script进行处理。对于参数个数不固定并且希望对每个命令参数 进行单独处理时则需要shift命令。使用shift可以将命令行位置参数依次移动位置, 即->, ->. 在移位之前的第一个位置参数在移位后将不在存在。
示例如下:
#!/bin/sh # #Filename: shifter
until [ $# -eq 0 ] do echo "Argument is and `expr $# - 1` argument(s) remain" shift done
$ shifter 1 2 3 4 Argument is 1 and 3 argument(s) remain Argument is 2 and 2 argument(s) remain Argument is 3 and 1 argument(s) remain Argument is 4 and 0 argument(s) remain $
使用shift时,每进行一次移位,$#减1,使用这一特性可以用until循环对每个参 数进行处理,如下示例中是一个求整数和的shell script:
#!/bin/sh # sumints - a program to sum a series of integers #
if [ $# -eq 0 ] then echo "Usage: sumints integer list" exit 1 fi
sum=0
until [ $# -eq 0 ] do sum=`expr $sum + ` shift done echo $sum
$ sh sumints 324 34 34 12 34 438 $
使用shift的另一个原因是Bourne Shell的位置参数变量为~, 因此通过位置变量 只能访问前9个参数。但这并不等于在命令行上最多只能输入9个参数。此时如果想访问 前9个参数之后的参数,就必须使用shift.
另外shift后可加整数进行一次多个移位,如:
shift 3
<4>. for循环 格式: for var in arg1 arg2 ... argn do command .... command done
示例: $ for letter in a b c d e; do echo $letter;done a b c d e
对当前目录下的所有文件操作: $ for i in * do if [ -f $i ] then echo "$i is a file" elif [ -d $i ] echo "$i is a directory" fi done
求命令行上所有整数之和: #!/bin/sh
sum=0
for INT in $* do sum=`expr $sum + $INT` done
echo $sum
<6> 从循环中退出: break和continue命令 break 立即退出循环 continue 忽略本循环中的其他命令,继续下一下循环
在shell编程中有时我们要用到进行无限循环的技巧,也就是说这种循环一直执行碰 到break或continue命令。这种无限循环通常是使用true或false命令开始的。UNIX 系统中的true总是返加0值,而false则返回非零值。如下所示:
#一直执行到程序执行了break或用户强行中断时才结束循环 while true do command .... command done
上面所示的循环也可以使用until false, 如下:
until false do command .... command done
在如下shell script中同时使用了continue,break以及case语句中的正规表达式用法:
#!/bin/sh # Interactive program to restore, backup, or unload # a directory
echo "Welcome to the menu driven Archive program"
while true do # Display a Menu echo echo "Make a Choice from the Menu below" echo _ echo "1 Restore Archive" echo "2 Backup directory" echo "3 Unload directory" echo "4 Quit" echo
# Read the user's selection echo -n "Enter Choice: "
read CHOICE
case $CHOICE in [1-3] ) echo # Read and validate the name of the directory
echo -n "What directory do you want? " read WORKDIR
if [ ! -d "$WORKDIR" ] then echo "Sorry, $WORKDIR is not a directory" continue fi
# Make the directory the current working directory cd $WORKDIR;;
4) :;; # :为空语句,不执行任何动作 *) echo "Sorry, $CHOICE is not a valid choice" continue esac
case "$CHOICE" in 1) echo "Restoring..." cpio -i 2) echo "Archiving..." ls | cpio -o >/dev/rmt/0h;;
3) echo "Unloading..." ls | cpio -o >/dev/rmt/0h;;
4) echo "Quitting" break;; esac
#Check for cpio errors
if [ $? -ne 0 ] then echo "A problem has occurred during the process" if [ $CHOICE = 3 ] then echo "The directory will not be erased" fi
echo "Please check the device and try again" continue else if [ $CHOICE = 3 ] then rm * fi fi done
(6)结构化编程:定义函数 同其他高级语言一样,shell也提供了函数功能。函数通常也称之为子过程(subroutine), 其定义格式如下:
funcname() { command ... command; #分号 }
定义函数之后,可以在shell中对此函数进行调用,使用函数定义可以将一个复杂的程序分 为多个可管理的程序段,如下所示:
# start program
setup () { command list ; }
do_data () { command list ; }
cleanup () { command list ; }
errors () { command list ; }
setup do_data cleanup # end program
技巧: . 在对函数命名时最好能使用有含义的名字,即函数名能够比较准确的描述函数所完成 的任务。 . 为了程序的维护方便,请尽可能使用注释
使用函数的另一个好处就是可以在一个程序中的不同地方执行相同的命令序列(函数), 如下所示:
iscontinue() { while true do echo -n "Continue?(Y/N)" read ANSWER
case $ANSWER in [Yy]) return 0;; [Nn]) return 1;; *) echo "Answer Y or N";; esac done }
这样可以在shell编程中调用iscontinue确定是否继续执行:
if iscontinue then continue else break fi
** shell函数与shell程序非常相似,但二者有一个非常重要的差别:shell程序是由子shell 执行的,而shell函数则是作为当前shell的一部分被执行的,因此在当前shell中可以改 变函数的定义。此外在任意shell(包括交互式的shell)中均可定义函数,如:
$ dir dir: not found $ dir () { ls -l ;} $ dir total 5875 -rw-r--r-- 1 hbwork 100 Nov 10 23:16 doc -rw-r--r-- 1 hbwork 2973806 Nov 10 23:47 ns40docs.zip -rw-r--r-- 1 hbwork 1715011 Nov 10 23:30 ns840usr.pdf -rw-r--r-- 1 hbwork 1273409 Sep 23 1998 radsol21b6.tar.Z -rw-r--r-- 1 hbwork 7526 Nov 10 23:47 wget-log -rw-r--r-- 1 hbwork 1748 Nov 13 21:51 wget-log.1 $ unset dir $ dir () { > echo "Permission Link Owner Group File_SZ LastAccess FileName" > echo "-----------------------------------------------------------" > ls -l $*; > }
$ dir Permission Link Owner Group File_SZ LastAccess FileName ----------------------------------------------------------- total 5875 -rw-r--r-- 1 hbwork 100 Nov 10 23:16 doc -rw-r--r-- 1 hbwork 2973806 Nov 10 23:47 ns40docs.zip -rw-r--r-- 1 hbwork 1715011 Nov 10 23:30 ns840usr.pdf -rw-r--r-- 1 hbwork 1273409 Sep 23 1998 radsol21b6.tar.Z -rw-r--r-- 1 hbwork 7526 Nov 10 23:47 wget-log -rw-r--r-- 1 hbwork 1748 Nov 13 21:51 wget-log.1
通常情况下,shell script是在子shell中执行的,困此在此子shell中对变量所作的 修改对父shell不起作用。点(.) 命令使用shell在不创建子shell而由当前shell读取 并执行一个shell script, 可以通过这种方式来定义函数及变量。此外点(.)命令最 常用的功能就是通过读取.profile来重新配置初始化login变量。如下所示:
$ . .profile (csh相对于.命令的是source命令).
(7)使用And/Or结构进行有条件的命令执行 <1> And , 仅当第一个命令成功时才有执行后一个命令,如同在逻辑与表达式中如果前面的 结果为真时才有必要继续运算,否则结果肯定为假。
格式如下:
command1 && command2
例:rm $TEMPDIR/* && echo "File successfully removed"
等价于
if rm $TEMPDIR/* then echo "File successfully removed" fi
<2>Or, 与AND相反,仅当前一个命令执行出错时才执行后一条命令
例: rm $TEMPDIR/* || echo "File not removed"
等价与:
if rm $TEMPDIR/* then command else echo "File not removed" fi
<3> 混合命令条件执行 command1 && command2 && command3 当command1, command2成功时才执行command3
command1 && command2 || comamnd3 仅当command1成功,command2失败时才执行command3
当然可以根据自己的需要进行多种条件命令的组合,在此不多讲述。
(8) 使用getopts命令读取unix格式选项 UNIX格式选项指如下格式的命令行参数: command -options parameters
使用格式: getopts option_string variable
具体使用方法请参考getopts的在线文档(man getopts).
示例如下:
#newdate if [ $# -lt 1 ] then date else while getopts mdyDHMSTjJwahr OPTION do case $OPTION in m) date '+%m ';; # Month of Year d) date '+%d ';; # Day of Month y) date '+%y ';; # Year D) date '+%D ';; # MM/DD/YY H) date '+%H ';; # Hour M) date '+%M ';; # Minute S) date '+%S ';; # Second T) date '+%T ';; # HH:MM:SS j) date '+%j ';; # day of year J) date '+%y%j ';;# 5 digit Julian date w) date '+%w ';; # Day of the Week a) date '+%a ';; # Day abbreviation h) date '+%h ';; # Month abbreviation r) date '+%r ';; # AM-PM time \?) echo "Invalid option $OPTION";; esac done fi
$ newdate -J 94031 $ newdate -a -h -d Mon Jan 31 $ newdate -ahd Mon Jan 31 $
示例程序:复制程序
# Syntax: duplicate [-c integer] [-v] filename # where integer is the number of duplicate copies # and -v is the verbose option
COPIES=1 VERBOSE=N
while getopts vc: OPTION do case $OPTION in c) COPIES=$OPTARG;; v) VERBOSE=Y;; \?) echo "Illegal Option" exit 1;; esac done
if [ $OPTIND -gt $# ] then echo "No file name specified" exit 2 fi
shift `expr $OPTIND -1`
FILE= COPY=0
while [ $COPIES -gt $COPY ] do COPY=`expr $COPY + 1` cp $FILE ${FILE}${COPY} if [ VERBOSE = Y ] then echo ${FILE}${COPY} fi done
$ duplicate -v fileA fileA1 $ duplicate -c 3 -v fileB fileB1 fileB2 fileB3
4. Shell的定制 通常使用shell的定制来控制用户自己的环境,比如改变shell的外观(提示符)以及增强 自己的命令。
(1)通常环境变量来定制shell 通常改变环境变量可以定制shell的工作环境。shell在处理信息时会参考这些环境变量 ,改变环境变量的值在一定程度上改变shell的操作方式,比如改变命令行提示符。
.使用IFS增加命令行分隔符 默认状态下shell的分隔符为空格、制表符及换行符,但可以通过改变IFS的值加入自己 的分隔符。如下所示:
$ IFS=":" $ echo:Hello:my:Friend Hello my Friend
(2)加入自己的命令及函数 如下程序: #Directory and Prompt change program #Syntax: chdir directory
if [ ! -d "" ] then echo " is not a directory" exit 1 fi
cd PS1=`pwd`$ export PS1
$ chdir /usr/home/teresa $
但此程序在执行时系统提示符并不会改变,因为此程序是在子shell中执行的。因此其变量 对当前shell并无影响,要想对当前shell起作用,最好是将此作为函数写在自己的.profile中 或建立自己的个人函数文件.persfuncs
#Personal function file persfuncs
chdir() { #Directory and Prompt change program #Syntax: chdir directory if [ ! -d "" ] then echo " is not a directory" exit 1 fi
cd PS1=`pwd`$ export PS1; }
再执行: $ . .persfuncs $ chdir temp /home/hbbwork/temp$
也可在自己的.profile文件中用 . .persfuncs调用.persfuncs.
说明:在bash/tcsh中已经使用别名,相对而言别名比此方法更为方便。
5. 有关shell的专门讨论 (1)shell程序的调试 切记:程序员(人)总是会犯错误的,而计算机是不会错的。 使用-x进行跟踪执行,执行并显示每一条指令。
(2)命令组 用小括号将一组命令括起来,则这些命令会由子shell来完成;而{command_list;}则在当 前shell中执行。这两者的主要区别在于其对shell变量的影响,子shell执行的命令不会 影响当前shell中的变量。
$ NUMBER=2 $ (A=2;B=2;NUMBER=`expr $A + $B`; echo $NUMBER) 4 $ echo $NUMBER 2 $ { A=2;B=2;NUMBER=`expr $A + $B`; echo $NUMBER; } 4 $ echo $NUMBER 4
总结: 在本章中讲述了Bourne Shell的基本知识,使用shell变量,shell script基础。这些概念 对于理解学习Korn Shell, csh以及其他script编程都是非常有用的。
很多OS都有不少语言及一些script功能,但很少有象UNIX SHELL这样灵活强大的script脚 本语言能力。
对于系统管理员或程序员来说,熟练地使用shell script将对日常工作(系统维护及管理) 非常有用,如果你想作一个合格的系统管理员,强烈建议你进一步深入的了解和使用 shell.
另外,对于系统管理员来说,PERL也是一个必不可少的script编程语言,尤其是对于处理 文本格式的各种文件,PERL具有shell, awk, sed, grep等的功能,但使用起来更为灵活, 功能也更强大。大家可以参考“Perl By Examples"来学习和使用PERL。
|