------------------------------------------------------------------------------ 
Bourne Shell 
  
介绍:Bourne Shell 基础及其他很多有用的特性,shell编程及组织。 
  
主要内容: 
.shell基础      基本介绍,环境,选项,特殊字符 
.shell变量      用户定义变量,环境变量,位置变量(shell 参数) 
.shell script编程 
        条件测试,循环及重复控制 
.shell定制 
  
1.shell基础知识 
  作者:Stephen Bourne 在Bell实验室开发 
  建议:man sh  查看相关UNIX上的改进或特性 
  
(1)shell提示符及其环境 
   /etc/passwd文件 
   提示符:$ 
   /etc/profile $HOME/.profile 
(2)shell执行选项 
   -n   测试shell script语法结构,只读取shell script但不执行 
   -x   进入跟踪方式,显示所执行的每一条命令,用于调度 
   -a   Tag all variables for export 
   -c "string"  从strings中读取命令 
   -e   非交互方式 
   -f   关闭shell文件名产生功能 
   -h   locate and remember functions as defind 
   -i   交互方式 
   -k   从环境变量中读取命令的参数 
   -r   限制方式 
   -s   从标准输入读取命令 
   -t   执行命令后退出(shell exits) 
   -u   在替换中如使用未定义变量为错误 
   -v   verbose,显示shell输入行 
  
   这些选项可以联合使用,但有些显然相互冲突,如-e和-i. 
  
(3)受限制shell(Restircted Shell) 
    sh -r 或 /bin/rsh 
  
    不能执行如下操作:cd, 更改PATH,指定全路径名,输出重定向,因此可以提供一个较好的控制和安全机制。通常rsh用于应用型用户及拨号用户,这些用户通常是看不到提 
示符的。通常受限制用户的主目录是不可写的。 
  
    不足:如果用户可以调用sh,则rsh的限制将不在起作用,事实上如果用户在vi及more 
        程序中调用shell,而这时rsh的限制将不再起作用。 
  
(4)用set改变 shell选项 
   用户可以在$提示符下用set命令来设置或取消shell的选项。使用-设置选项,+取消相应 
   选项,大多数UNIX系统允许a,e,f,h,k,n,u,v和x的开关设置/取消。 
  
   set -xv 
        启动跟踪方式;显示所有的命令及替换,同样显示输入。 
   set -tu 
        关闭在替换时对未定义变量的检查。 
  
   使用echo $-显示所有已设置的shell选项。 
  
  
(5)用户启动文件 .profile 
        PATH=$PATH:/usr/loacl/bin; export PATH 
  
(6)shell环境变量 
        CDPATH  用于cd命令的查找路径 
        HOME    /etc/passwd文件中列出的用户主目录 
        IFS     Internal Field Separator,默认为空格,tab及换行符 
        MAIL    /var/mail/$USERNAME     mail等程序使用 
        PATH 
        PS1,PS2        默认提示符($)及换行提示符(>) 
        TERM    终端类型,常用的有vt100,ansi,vt200,xterm等 
  
        示例:$PS1="test:";export PS1 
              test: PS1="\$";export PS1 
              $echo $MAIL 
              /var/mail/username 
(7)保留字符及其含义 
$      shell变量名的开始,如$var 
   |    管道,将标准输出转到下一个命令的标准输入 
   #    注释开始 
   &    在后台执行一个进程 
   ?   匹配一个字符 
   *    匹配0到多个字符(与DOS不同,可在文件名中间使用,并且含.) 
   $-   使用set及执行时传递给shell的标志位 
   $!   最后一个子进程的进程号 
   $#   传递给shell script的参数个数 
   $*   传递给shell script的参数 
   $@   所有参数,个别的用双引号括起来 
   $?   上一个命令的返回代码 
   $0   当前shell的名字 
   $n    (n:1-) 位置参数 
   $$   进程标识号(Process Identifier Number, PID) 
   >file        输出重定向 
   <file        输入重定向 
   `command`    命令替换,如    filename=`basename /usr/local/bin/tcsh` 
   >>fiile      输出重定向,append 
  
   转义符及单引号: 
        $echo "$HOME $PATH" 
        /home/hbwork /opt/kde/bin:/usr/local/bin:/bin:/usr/bin:/usr/X11R6/bin: 
        $echo '$HOME $PATH' 
        $HOME $PATH 
        $echo \$HOME $PATH 
        $HOME /opt/kde/bin:/usr/local/bin:/bin:/usr/bin:/usr/X11R6/bin:/home/hbw 
ork/bin 
  
   其他: 
        $dir=ls 
        $$dir 
        $alias dir ls 
        $dir 
  
        ls > filelist 
        ls >> filelist 
        wc -l < filelist 
        wc -l filelist 
        sleep 5; echo 5 seconds reaches; ls -l 
        ps ax |egrep inetd 
        find / -name core -exec rm {} \; & 
        filename=`date "+%Y%m%d"`.log 
  
2. shell变量 
  变量:代表某些值的符号,如$HOME,cd命令查找$HOME,在计算机语言中可以使用变量可以 
        进行多种运算和控制。 
  
  Bourne Shell有如下四种变量: 
        .用户自定义变量 
        .位置变量即 shell script之参数 
        .预定义变量(特殊变量) 
        .环境变量(参考shell定制部分) 
(1)用户自定义变量(数据的存储) 
        $ COUNT=1 
        $ NAME="He Binwu" 
  
     技巧:因为大部分UNIX命令使用小写字符,因此在shell编程中通常使用全大写变量, 
        当然这并不是强制性的,但使用大写字符可以在编程中方便地识别变量。 
  
     变量的调用:在变量前加$ 
        $ echo $HOME 
        /home/hbwork 
        $ WEEK=Satur 
        $ echo Today is $WEEKday 
        Today is 
        $echo Today is ${WEEK}day 
        Today is Saturday 
  
     Shell变量赋值从右从左进行(Linux Shell/bash从左向右赋值!) 
     $ X=$Y Y=y 
     $ echo $X 
     y 
     $ Z=z Y=$Z 
     $ echo $Y 
  
     $ 
  
     使用unset命令删除变量的赋值 
     $ Z=hello 
     $ echo $Z 
     hello 
     $ unset Z 
     $ echo $Z 
  
     $ 
  
     有条件的命令替换 
        在Bourne Shell中可以使变量替换在特定条件下执行,即有条件的环境变量替换。 
        这种变量替换总是用大括号括起来的。 
  
        .设置变量的默认值 
            在变量未赋值之前其值为空。Bourne Shell允许对变量设置默认值,其格式如 
下: 
            ${variable:-defaultvalue} 
            例: 
                $ echo Hello $UNAME 
                Hello 
                $ echo Hello ${UNAME:-there} 
                Hello there 
                $ echo $UNAME   #变量值并未发生变化 
  
                $ UNAME=hbwork 
                $ echo Hello ${UNAME:-there} 
                Hello hbwork 
                $ 
        .另一种情况:改变变量的值,格式如下: 
            ${variable:=value} 
  
            例: 
                $ echo Hello $UNAME 
                Hello 
                $ echo Hello ${UNAME:=there} 
                Hello there 
                $ echo $UNAME   #变量值并未发生变化 
                there 
                $ 
        .变量替换中使用命令替换 
                $USERDIR=${$MYDIR:-`pwd`} 
  
        .在变量已赋值时进行替换  ${variable:+value} 
        .带有错误检查的有条件变量替换 
          ${variable:?value} 
          例: 
          $ UNAME= 
          $ echo ${UNAME:?"UNAME has not been set"} 
          UNAME: UNAME has not been set 
          $ echo ${UNAME:?} 
          UNAME: parameter null or not set 
  
  (2)位置变量(Shell参数) 
  在shell script中位置参数可用$1..$9表示,$0表示内容通常为当前执行程序的文件名。 
  .防止变量值被替换 readonly variable 
  .使用export命令输出变量,使得变量对子shell可用,当shell执行一下程序时,shell 
   将为其设置一个新的环境让其执行,这称之了subshell.  在Bourne Shell中变量通常 
   被认为是本地变量,也就是说在对其赋值之外的shell环境之外是不认识此变量的。使 
   用export对subshell可用。 
  
   例:对变量PS1的export操作,shell的提示符将发生变化。 
   $ PS1=`hostname`$ 
        peony$sh 
        $ echo $PS1 
        $  <-输出结果 
        $ exit 
        peony$export PS1 
        peony$sh 
        peony$ echo $PS1 
        peony$  <-输出结果 
        peony$ 
  
  
3.Shell Script编程 
目的:使用UNIX所提供的最常用工具来完成所需复杂任务的强大功能。 
  
(1)最简单的Shell 编程 
   $ls -R / |grep myname |more 
  
   每天数据的备份: 
   $ cd /usr/yourname; ls * |cpio -o > /dev/rmt/0h 
  
   书写程序的目的是一次编程,多次使用(执行)! 
  
   $ cat > backup.sh 
   cd /home/hbwork 
   ls * | cpio -o > /dev/rmt/0h 
   ^D 
  
   执行: 
   $ sh backup.sh 
  
   或: 
   $ chmod +x backup.sh 
   $ ./backup.sh 
  
   技巧:在shell script中加入必要的注释,以便以后阅读及维护。 
  
(2)shell是一个(编程)语言 
  同传统的编程语言一样,shell提供了很多特性,这些特性可以使你的shell script 
  编程更为有用,如:数据变量、参数传递、判断、流程控制、数据输入和输出,子 
  程序及以中断处理等。 
  
  . 在shell编程中使用数据变量可以将程序变量更为通用,如在上面backup.sh中: 
    cd $WORKDIR 
    ls * | cpio -o > /dev/rmt/0h 
  
  . Shell编程中的注释以#开头 
  . 对shell变量进行数字运算,使用expr命令 
        expr integer operator integer 
        其中operator为+ - * / %, 但对*的使用要用转义符\,如: 
        $expr 4 \* 5 
        20 
        $int=`expr 5 + 7` 
        $echo $int 
        12 
  
(3)Shell编程的参数传递, 可通过命令行参数以及交互式输入变量(read) 
  
     restoreall.sh 对backup.sh程序的备份磁带进行恢复 
        $cat > restoreall.sh 
        cd $WORKDIR 
        cpio -i < /dev/rmt/0h 
        ^D 
     restore1.sh:只能恢复一个文件 
        #restore1 --program to restore a single file 
        cd $WORKDIR 
        cpio -i $i < /dev/rmt/0h 
  
        $restore1 file1 
  
        恢复多个文件restoreany : 
        #restoreany --program to restore a single file 
        cd $WORKDIR 
        cpio -i $* < /dev/rmt/0h 
  
        $ restoreany file1 file2 file3 
  
  
(4)条件判断 
   . if-then语句,格式如下: 
        if command_1 
        then 
                command_2 
                command_3 
        fi 
        command_4 
  
   在if-then语句中使用了命令返回码$?,即当command_1执行成功时才执行command_2和command_3,而command_4总是执行. 
  
   示例程序unload: 在备份成功时删除原始文件,带有错误检查 
  
        cd $1 
        #备份时未考虑不成功的情况! 
        ls -a | cpio -o > /dev/rmt/0h 
        rm -rf * 
  
        改进如下: 
  
        #当使用了管道命令时,管理命令的返回代码为最后一个命令的返回代码 
        if ls -a | cpio -o > /dev/rmt/0h 
        then 
                rm -rf * 
        fi 
  
   . if-then-else语句 
        if command_1 
        then 
                command_2 
        else 
                command_3 
        fi 
  
        技巧: 由于shell对命令中的多余的空格不作任何处理,一个好的程序员会用这一特性对自己的程序采用统一的缩进格式,以增强自己程序的可读性. 
  
    . 使用test命令进行进行条件测试 
      格式: test conditions 
  
      test在以下四种情况下使用: a. 字符比较 b.两个整数值的比较 
                c. 文件操作,如文件是否存在及文件的状态等 
                d. 逻辑操作,可以进行and/or,与其他条件联合使用 
  
      a. 测试字符数据: shell变量通常民政部下均作为字符变量 
        str1 = str2     二者相长,相同 
        str1 != str2    不同 
        -n string       string不为空(长度不为零) 
        -z string       string为空 
        string          string不为空 
  
        例: 
                $ str1=abcd     #在含有空格时必须用引号括起来 
                $ test $str1=abcd 
                $ echo $? 
                0 
                $ str1="abcd " 
                $ test $str1=abcd 
                $ echo $? 
                1 
        Note: 在test处理含有空格的变量时最好用引号将变量括起来,否则会出现错误的结果, 因为shell在处理命令行时将会去掉多余的空格,而用引号括起来则可以防止 
shell去掉这些空格. 
              例: 
                $ str1="    " 
                $ test $str1 
                $ echo $? 
                1 
                $ test "$str1" 
                $ echo $? 
                0 
                $ test -n $str1 
                test: argument expected 
                $ test -n "$str1" 
                $ echo $? 
                0 
                $ 
  
      b. 整数测试: test与expr相同,可以将字符型变量转换为整数进行操作,expr进行 
         整数的算术运算,而test则进行逻辑运算. 
  
         表达式                 说明 
         --------------------------------------- 
         int1 -eq int2          相等? 
         int1 -ne int2          不等? 
         int1 -gt int2          int1 > int2 ? 
         int1 -ge int2          int1 >= int2 ? 
         int1 -lt int2          int1 < int2 ? 
         int1 -le int2          int1 <= int2 ? 
  
         例: 
                $ int1=1234 
                $ int2=01234 
                $ test $int1 -eq $int2 
                $ echo $? 
                0 
  
      c. 文件测试:检查文件状态如存在及读写权限等 
  
         -r filename     用户对文件filename有读权限? 
         -w filename     用户对文件filename有写权限? 
         -x filename     用户对文件filename有可执行权限? 
         -f filename     文件filename为普通文件? 
         -d filename     文件filename为目录? 
         -c filename     文件filename为字符设备文件? 
         -b filename     文件filename为块设备文件? 
         -s filename     文件filename大小不为零? 
         -t fnumb        与文件描述符fnumb(默认值为1)相关的设备是一个终端设备? 
  
      d. 测试条件之否定,使用! 
        例: 
                $ cat /dev/null > empty 
                $ test -r empty 
                $ echo $? 
                0 
                $ test -s empty 
                1 
                $ test ! -s empty 
                $ echo $? 
                0 
      e. 测试条件之逻辑运算 
        -a      And 
        -o      Or 
  
        例: $ test -r empty -a -s empty 
            $ echo $? 
            1 
      f. 进行test测试的标准方法 
         因为test命令在 shell编程中占有很重要的地位,为了使shell能同其他编程语一样便于阅读和组织, Bourne Shell在使用test测试时使用了另一种方法:用方括号将整个test测试括起来: 
  
         $ int1=4 
         $ [ $int1 -gt 2 ] 
         $ echo $? 
         0 
  
         例: 重写unload程序,使用test测试 
         #!/bin/sh 
         #unload - program to backup and remove files 
         #syntax: unload directory 
  
         #check arguments 
         if [ $# -ne 1 ] 
         then 
                echo "usage: $0 directory" 
                exit 1 
         fi 
  
         #check for valid directory name 
         if [ ! -d "$1" ] 
         then 
                echo "$1 is not a directory" 
                exit 2 
         fi 
  
         cd $1 
  
         ls -a | cpio -o > /dev/rmt/0h 
  
         if [ $? -eq 0 ] 
         then 
                rm -rf * 
         else 
                echo "A problem has occured in creating backup" 
                echo "The directory will not be ereased" 
                echo "Please check the backup device" 
                exit 3 
         fi 
         # end of unload 
  
         在如上示例中出现了exit, exit有两个作用:一是停止程序中其他命令的执行,二 
是 
         设置程序的退出状态 
  
      g. if嵌套及elif结构 
        if command 
        then 
                command 
        else 
                if command 
                then 
                        command 
                else 
                        if command 
                        then 
                                command 
                        fi 

                fi 
        fi 
  
        改进:使用elif结构 
        if command 
        then 
                command 
        elif    command 
        then 
                command 
        elif    command 
        then 
                command 
        fi 
  
        elif结构同if结构类似,但结构更清淅,其执行结果完全相同.
posted on 2010-10-25 14:20  一个人的天空@  阅读(1741)  评论(0编辑  收藏  举报