测试路漫漫, 吾将上下求索. 两年多开发经验, 六年整测试经验. 比较熟悉web自动化测试, especially in Ruby & Watir. 正在探讨Agile 测试, Junit testing in Agile. 联系方式:myfengliu@hotmail.com

转贴Linux Shell编程(基础教程)

上一篇 / 下一篇  2010-07-20 08:06:14 / 个人分类:linux shell

本文网址:http://bbs.bitscn.com/72875复制51Testing软件测试网j5Ad G7\1Y'YAs

j3i9UOqR)A\8j01. Linux 脚本编写基础
`-Wi@L01.1 语法基本介绍51Testing软件测试网Qe9O^G6YB+T&F&JE
1.1.1 开头51Testing软件测试网7qOO.VM*R
       程序必须以下面的行开始(必须放在文件的第一行):51Testing软件测试网1HT2E@B8v
       #!/bin/sh51Testing软件测试网;o2G U0yH
    符号#!用来告诉系统它后面的参数是用来执行该文件的程序。在这个例子中我们使用/bin/sh来执行程序。  当编辑好脚本时,如果要执行该脚本,还必须使其可执行。
K:c2EF5e;]#h.u0    要使脚本可执行:   编译 chmod +x filename 这样才能用./filename 来运行
LLzE?$K;m`)M01.1.2 注释
r O8b j&x(cS9v+q6N+H0  在进行shell编程时,以#开头的句子表示注释,直到这一行的结束。我们真诚地建议您在程序中使用注释。如果您使用了注释,那么即使相当长的时间内没有使用该脚本,您也能在很短的时间内明白该脚本的作用及工作原理。51Testing软件测试网c4aS!`'V O_[t
1.1.3 变量
D'G5~E v5p0  在其他编程语言中您必须使用变量。在shell编程中,所有的变量都由字符串组成,并且您不需要对变量进行声明。要赋值给一个变量,您可以这样写:51Testing软件测试网z6cc ^"{-IX
        #!/bin/sh51Testing软件测试网'jX'mXu e1a
        #对变量赋值:
2X4B!M0A,@M{0        a="hello world"
EfK/eO7rB/~G S,Ng0        # 现在打印变量a的内容:51Testing软件测试网;FG4L'o uX0b
        echo "A is:"
Xl7M x*Y,ZM a Lp0        echo $a
alb1a_({b v#k0       有时候变量名很容易与其他文字混淆,比如:51Testing软件测试网6} C Q-`&~@8E
        num=2
'q"AF)]u w2s0        echo "this is the $numnd"
_QW"|l}4B0       这并不会打印出"this is the 2nd",而仅仅打印"this is the ",因为shell会去搜索变量numnd的值,但是这个变量时没有值的。可以使用花括号来告诉shell我们要打印的是num变量:51Testing软件测试网/pm0R&vE7d1zp2Z
        num=2
F)dI:l8l.gw-g]5{0        echo "this is the ${num}nd"51Testing软件测试网B {*e2UbJ5@6h
        这将打印: this is the 2nd51Testing软件测试网 K(H,o`OeL9qG
1.1.4 环境变量
d-rp`9vz6G!O0       由export关键字处理过的变量叫做环境变量。我们不对环境变量进行讨论,因为通常情况下仅仅在登录51Testing软件测试网/x2g"qyy
脚本中使用环境变量。
bH8H$rq&XG'\1v01.1.5 Shell命令和流程控制51Testing软件测试网_yTUoX
       在shell脚本中可以使用三类命令:
feHM/?0       1)Unix 命令:51Testing软件测试网4]~3g4ywmf%TW'T{ Xs
      虽然在shell脚本中可以使用任意的unix命令,但是还是由一些相对更常用的命令。这些命令通常是用来进行文件和文字操作的。51Testing软件测试网5LEN(]4xc%~_
        常用命令语法及功能
3m&sw'N)TbK0  echo "some text": 将文字内容打印在屏幕上
"v$a/uz2J ?8f0  ls: 文件列表
9w7ZXy1\;dO?0  wc –l filewc -w filewc -c file: 计算文件行数计算文件中的单词数计算文件中的字符数
&?lpSVf{#v |2\9F0  cp sourcefile destfile: 文件拷贝
A2\%o3s#s#Wa5h0  mv oldname newname : 重命名文件或移动文件51Testing软件测试网:}6p:@Q&?1_ij
  rm file: 删除文件51Testing软件测试网0zm?0DUtf'I~
  grep 'pattern' file: 在文件内搜索字符串比如:grep 'searchstring' file.txt
6q7t0@-U5iFvW0  cut -b colnum file: 指定欲显示的文件内容范围,并将它们输出到标准输出设备比如:输出每行第5个到第9个字符cut -b5-9 file.txt千万不要和cat命令混淆,这是两个完全不同的命令51Testing软件测试网 K9W aD5q
  cat file.txt: 输出文件内容到标准输出设备(屏幕)上51Testing软件测试网D"e\ s'b4k|^V
  file somefile: 得到文件类型
#YKP-K o|B|0  read var: 提示用户输入,并将输入赋值给变量
}|;iu/Ox2x0  sort file.txt: 对file.txt文件中的行进行排序51Testing软件测试网3`(|.Ua9`'S3qg7m\8z
  uniq: 删除文本文件中出现的行列比如: sort file.txt | uniq51Testing软件测试网x"u{^9sR J5K
  expr: 进行数学运算Example: add 2 and 3expr 2 "+" 3
NqDc HG&h b4F0  find: 搜索文件比如:根据文件名搜索find . -name filename -print51Testing软件测试网!pRQ!s I8U` `;W
  tee: 将数据输出到标准输出设备(屏幕) 和文件比如:somecommand | tee outfile51Testing软件测试网Z~$B/ydo
  basename file: 返回不包含路径的文件名比如: basename /bin/tux将返回 tux51Testing软件测试网9M9bY:s1s
  dirname file: 返回文件所在路径比如:dirname /bin/tux将返回 /bin51Testing软件测试网N`(z C)| M
  head file: 打印文本文件开头几行51Testing软件测试网Q4j1S/W b:f!S7o
  tail file : 打印文本文件末尾几行
VWB Jg-m+Z&]0  sed: Sed是一个基本的查找替换程序。可以从标准输入(比如命令管道)读入文本,并将结果输出到标准输出(屏幕)。该命令采用正则表达式(见参考)进行搜索。 不要和shell中的通配符相混淆。比如:将linuxfocus 替换为LinuxFocus :cat text.file | sed 's/linuxfocus/LinuxFocus/' > newtext.file
ha:h#f:A$H3a'e0  awk: awk 用来从文本文件中提取字段。缺省地,字段分割符是空格,可以使用-F指定其他分割符。51Testing软件测试网0Bh(h X lv,_0FH
cat file.txt | awk -F, '{print $1 "," $3 }'这里我们使用,作为字段分割符,同时打印第一个和第三个字段。如果该文件内容如下: Adam Bor, 34, IndiaKerry Miller, 22, USA51Testing软件测试网P F@ NFtH
        命令输出结果为:Adam Bor, IndiaKerry Miller, USA51Testing软件测试网.[,E.D.y.Zu(}V$R
        2) 概念: 管道, 重定向和 backtick51Testing软件测试网a'EJ MS3D
  这些不是系统命令,但是他们真的很重要。
(DZ ?.qg,u$f/Q0  管道 (|) 将一个命令的输出作为另外一个命令的输入。
_n!Q,RLbAO0                 grep "hello" file.txt | wc -l51Testing软件测试网["p6]'m2@t
  在file.txt中搜索包含有”hello”的行并计算其行数。
4YWLc#Z V0  在这里grep命令的输出作为wc命令的输入。当然您可以使用多个命令。51Testing软件测试网9Afc"UV\q p%h)T
  重定向:将命令的结果输出到文件,而不是标准输出(屏幕)。
_H Z d)l0      > 写入文件并覆盖旧文件
+f#fV*c.e2J9sVuG0      >> 加到文件的尾部,保留旧文件内容。51Testing软件测试网i.g4}3ij;Sh-U J.G
        反短斜线
:Fx;VFo+O0     使用反短斜线可以将一个命令的输出作为另外一个命令的一个命令行参数。
AQ]HCv1^!f0         命令:51Testing软件测试网MTY C\!d"d ip
           find . -mtime -1 -type f -print
3c%J6liGK(QA2mu7l(a0     用来查找过去24小时(-mtime –2则表示过去48小时)内修改过的文件。如果您想将所有查找到的文件打一个包,则可以使用以下脚本:51Testing软件测试网K7c+DN(K@:w9ZuE7G
             #!/bin/sh51Testing软件测试网A _[Z#X$x0J
             # The ticks are backticks (`) not normal quotes ('):
H2\*oI]8hb!?0t0              tar -zcvf lastmod.tar.gz `find . -mtime -1 -type f -print`51Testing软件测试网q!E*wg5} Q
       3) 流程控制
#R-k @@NI.H0        1.if51Testing软件测试网[G&~n [)]/n/\
    "if" 表达式 如果条件为真则执行then后面的部分:51Testing软件测试网IHl:q+?
            if ....; then
q"sY*bbr&`W0           ....
[0RgKi0           elif ....; then
6PM)}DSI0           ....
"DkD'zJ:x'q0           else
S*]5}8T2C"sq0           ....51Testing软件测试网S,dww{T
           fi51Testing软件测试网 ib&{m/}I;S
           大多数情况下,可以使用测试命令来对条件进行测试。比如可以比较字符串、判断文件是否存在及是否可读等等…
/@8Y,C,WM7J4i0    通常用" [ ] "来表示条件测试。注意这里的空格很重要。要确保方括号的空格。
[$?7d%pbs"O0                 [ -f "somefile" ] :判断是否是一个文件
9Z6g(f2e Q/Ty0                 [ -x "/bin/ls" ] :判断/bin/ls是否存在并有可执行权限51Testing软件测试网"{{m2Kd%@
                 [ -n "$var" ] :判断$var变量是否有值
'[[hK~K4Na4l0                 [ "$a" = "$b" ] :判断$a和$b是否相等51Testing软件测试网z SKTBH
    执行man test可以查看所有测试表达式可以比较和判断的类型。51Testing软件测试网h7D ?:K]WV~
   直接执行以下脚本:
Q]qM'y&is8Nk0               #!/bin/sh
Y)`&XD|q0               if [ "$SHELL" = "/bin/bash" ]; then51Testing软件测试网K+O`yDu+P
                    echo "your login shell is the bash (bourne again shell)"
0Z k@3\h|DJ@0               else
{ qC tt,{;@6N0                    echo "your login shell is not bash but $SHELL"
$E V9DRS8v0               fi
gW%huAEf0   变量$SHELL包含了登录shell的名称,我们和/bin/bash进行了比较。
+G!CH:i taf w(~0        
9v'C Ir1X~9G&|2v"~0        快捷操作符51Testing软件测试网HD Z#O2u[(g
        熟悉C语言的朋友可能会很喜欢下面的表达式:51Testing软件测试网H~%@'tT ['rI:?.u
                  [ -f "/etc/shadow" ] && echo "This computer uses shadow passwors"
i\qz#e0  这里 && 就是一个快捷操作符,如果左边的表达式为真则执行右边的语句。您也可以认为是逻辑运算中的与操作。上例中表示如果/etc/shadow文件存在则打印” This computer uses shadow passwors”。同样或操作(||)在shell编程中也是可用的。这里有个例子:
5N8{ S\t0                 #!/bin/sh
#lqv!qZ z&N0                 mailfolder=/var/spool/mail/james
Rt |S0}e,L jm0                 [ -r "$mailfolder" ]' '{ echo "Can not read $mailfolder" ; exit 1; }
M7h!@X(t'q E0                 echo "$mailfolder has mail from:"
JJp([E'J0                 grep "^From " $mailfolder51Testing软件测试网@O\3d$h;I/cp
        该脚本首先判断mailfolder是否可读。如果可读则打印该文件中的"From" 一行。如果不可读则或操作生效,打印错误信息后脚本退出。这里有个问题,那就是我们必须有两个命令:51Testing软件测试网 Y)R mjQw I*w~u$q
  -打印错误信息51Testing软件测试网Xe Cj?
  -退出程序
5y&G$JBzI9RA?'b0  我们使用花括号以匿名函数的形式将两个命令放到一起作为一个命令使用。一般函数将在下文提及。51Testing软件测试网@t|,`} \*N1KK
  不用与和或操作符,我们也可以用if表达式作任何事情,但是使用与或操作符会更便利很多。51Testing软件测试网+z ?F!PQ

6K,NFJ7bV0         2.case51Testing软件测试网u-rh:hN+Y|B
               case :表达式可以用来匹配一个给定的字符串,而不是数字。
x;K5c J'HjcF0                      case ... in51Testing软件测试网F/G%k#}c4MGY
                            ...) do something here ;;51Testing软件测试网8G [Fr+S7Q/j
                      esac51Testing软件测试网Nz s$aGT,EU
         让我们看一个例子。 file命令可以辨别出一个给定文件的文件类型,比如:
'I$EIchOZ"~G2?-]0                     file lf.gz51Testing软件测试网V#X9]g Ai l3r
                   这将返回:51Testing软件测试网 L}[4P.[6D%\ E+\
                         lf.gz: gzip compressed data, deflated, original filename,
)R|(y0}mF7~0                         last modified: Mon Aug 27 23:09:18 2001, os: Unix51Testing软件测试网Q HkL9[9mXD
             我们利用这一点写了一个叫做smartzip的脚本,该脚本可以自动解压bzip2, gzip 和zip 类型的压缩文件:51Testing软件测试网*U~9x,F/\xK7u0p
               #!/bin/sh51Testing软件测试网Ew5N9@.R!do,V
               ftype=`file "$1"`
U0sv }%v0Y$I0               case "$ftype" in51Testing软件测试网/o)Cg4F@8Xk#m
                       "$1: Zip archive"*)
'P@} JV9L:nt.K*G)zK0                  unzip "$1" ;;
*Y:v"t3~&s0                         "$1: gzip compressed"*)51Testing软件测试网B*^^s)EE
                    gunzip "$1" ;;51Testing软件测试网lJ$yw,n ZZ h
                          "$1: bzip2 compressed"*)51Testing软件测试网Ge|q7a
                     bunzip2 "$1" ;;
H2HZz.AQ#}0                           *) echo "File $1 can not be uncompressed with smartzip";;
1\)o&l"d} ~ R&_0                esac51Testing软件测试网,~_df;U1[*v:j]$X{
         您可能注意到我们在这里使用了一个特殊的变量$1。该变量包含了传递给该程序的第一个参数值。51Testing软件测试网 ^*{G(q6Q
也就是说,当我们运行:51Testing软件测试网 i0o-]q/o
                    smartzip articles.zip
4C/\l `#@Q7dQ}0                    $1 就是字符串 articles.zip
*X7F9F] T E0                3. selsect51Testing软件测试网J:{|-m)S,l+q+d
                  select 表达式是一种bash的扩展应用,尤其擅长于交互式使用。用户可以从一组不同的值中进行选择。51Testing软件测试网?9Q&I7pLz3L
                    select var in ... ; do51Testing软件测试网 ^1]F-@g
                         break51Testing软件测试网6i+bh_WO"h5w
                            done51Testing软件测试网G$pf(Y%}d{q}
                            .... now $ var can be used ....51Testing软件测试网!hE C@$_R/C
                   下面是一个例子:
PR:c/{ydc,x0                         #!/bin/sh
4a8B`1c4iHr0                         echo "What is your favourite OS?"51Testing软件测试网6\ NY tT*]$A
                         select var in "Linux" "Gnu Hurd" "Free BSD" "Other"; do51Testing软件测试网Akj ^^
                    break
/k] }z)u0P%OO2Q+rd0                                done
-A b PRT3Mm0                           echo "You have selected $var"51Testing软件测试网C#S,t!_ {l&~$f'hG
              下面是该脚本运行的结果:51Testing软件测试网)cM_9n4QcZ
                               What is your favourite OS?
&t%Ja_/k"Op,p.i0                                    1) Linux51Testing软件测试网D[+gb-X;tp/p rC
                                    2) Gnu Hurd
U.GM!gK xUaH(g%h0                                    3) Free BSD
I)^j-u8K&Rq0                                    4) Other51Testing软件测试网3}^3IV5B$X7T;a? cXB3[
                                  #? 1
:n_OTi"E"S&pr0                                You have selected Linux
6R Ta'L y y6E0         注:var是个变量,可以换成其它的值。break用来跳出循环,如果没有break则一直循环下去。done与select对应。51Testing软件测试网@;U{B1w,ldz&f|w
                  4.loop
1wq F.i t0                      loop表达式:
F!@_2A/[0                            while ...; do
2D)?$FR4G*[7Ojn0                                      ....
1g\/h'A#@[5Q0                               done51Testing软件测试网WC$ti1fU.t}.C
             while-loop 将运行直到表达式测试为真。will run while the expression that we test for is true.51Testing软件测试网Nsxb)Y6^ n$w M
关键字"break" 用来跳出循环。而关键字”continue”用来不执行余下的部分而直接跳到下一个循环。51Testing软件测试网H2xF2j ~dd _I
  
2c-zB5R4i{m0                     for-loop表达式查看一个字符串列表 (字符串用空格分隔) 然后将其赋给一个变量:51Testing软件测试网:J,Q~,F dm
                          1,   for var in ....; do
.N(R.o+CI.d@2Qie+y0                                 ....
!a@;k;E:je4O&y0                                done51Testing软件测试网)MjO9[(IiqJo {5L
                       在下面的例子中,将分别打印ABC到屏幕上:51Testing软件测试网$P'f1VZ9{$UAl
                            #!/bin/sh51Testing软件测试网O0H[P2J!hS
                                    for var in A B C ; do
"f!TD6i8G!vV I _0                                 echo "var is $var"
Y!X!N|3f;K w;q0                                     done
@2E7vkE8@h0
rkGe6]_t2P2}0                           2,   for (( 条件一; 条件二; 条件三 );do
5S2j vRo'T%rX9WR0                                ...
.a1\;b&v b W4ww,Y/b0                                done
Tf$?LmU0                        例:51Testing软件测试网c_%jT1f7j@;cU\
                     51Testing软件测试网z;ieq4t#]vl
                   for ((i=1;i<10;i=$[$i+1]));do
^un {iw%H{0
+v*~-] T&f*n#j'_A0                             echo "a"
J,]*^A(L8h0D+jb0                          done51Testing软件测试网-HQ9j$vjQ7mIRo/T
输出:
I4W~#j J'I;|051Testing软件测试网k*L;V#kHv NN/M
a51Testing软件测试网(d)_ m!|d*@a,d"x9d
a
Yik?,AbQ;o)|"fk0a51Testing软件测试网 \ E`k_Tf0z+b_ w
a
sU bOZ](O"@ @0a51Testing软件测试网5f,ozDl[&G;G
a
,XTAGl7q0a51Testing软件测试网'V t _Y:K'Ykc
a51Testing软件测试网 ~ D j-a#]S
a
4[2apJ)ypB h0
ps3V*F?0o%y%l`0条件一:这可以看成是『初始值』,如上面的例子中,初始值是 i=1 啦!51Testing软件测试网?h:bZi*{
条件二:这可以看成是『符合值』,如上面的例子中,当 i<=100 的时候都是符合条件的!51Testing软件测试网 dR4\5x8y,?4b]6u#i
条件三:这可以看成是『步阶』!也就是说, i 每次都加一! 所以啦!上面的例子是说:由 i=1 开始到 i<= 100 ,每次 i 都加一来执行底下的程序段(就是 s=s+i ),当 i >100 (也就是 i=101 )就跳出这一段程序段!怎样!不难吧!51Testing软件测试网th!OT/Wr

#PxF!]6M%E/l/qkX0
0g7H~;] t2E7_051Testing软件测试网d,lJ] s;yK!T

5\8F!Cl,J#O(v%?9d0                        下面是一个更为有用的脚本showrpm,其功能是打印一些RPM包的统计信息:51Testing软件测试网Kz G4h"In
                          #!/bin/sh
q3d7R q*P`4v.` H0                          # list a content summary of a number of RPM packages51Testing软件测试网)}C'c K*fwR6@
                          # USAGE: showrpm rpmfile1 rpmfile2 ...51Testing软件测试网W7d.Y#a"gl}K&|
                          # EXAMPLE: showrpm /cdrom/RedHat/RPMS/*.rpm
l6??,hWI5\`O0                            for rpmpackage in $*; do
I#Re}Z(l#]0                         if [ -r "$rpmpackage" ];then
N Y%^Zr*F0                         echo "=============== $rpmpackage =============="51Testing软件测试网Q5i gVx0~BD
                          rpm -qi -p $rpmpackage51Testing软件测试网 Wrf$L%n
                          else
Y F{(zr5U9wS$V{1n0                           echo "ERROR: cannot read file $rpmpackage"51Testing软件测试网of c2v O
                         fi
"Q2vp8\za0                              done51Testing软件测试网:Jo9vr1b,ry q#cU:H
                   这里出现了第二个特殊的变量$*,该变量包含了所有输入的命令行参数值。51Testing软件测试网4e"U$F/`[o
                        如果您运行showrpm openssh.rpm w3m.rpm webgrep.rpm
[:Au9YY0                                   此时 $* 包含了 3 个字符串,即openssh.rpm, w3m.rpm and webgrep.rpm.51Testing软件测试网#MG7E4TgKV

m n?]r4h L W*CXm0
nO#W!SJ;d0
{+OUXT/HW/Z0『until:直到条件相同的时候才离开程序』;51Testing软件测试网mht'[8fgen#S
『while:当条件相同的时候,就继续做!』  
V+^AY:M~+H/b`1z0
s$F\3]qH2R!ow0until [ condition1 ] && { || } [ condition2 ] ...51Testing软件测试网P~S"tb"Uh
51Testing软件测试网6|2f0{8d6`

/U}~ Gq+CcK Z0
~ G h^9u2Ly x6J051Testing软件测试网5`dh5X-p _ ^8M B
                 5. 引号
s `D E9@+[0                   在向程序传递任何参数之前,程序会扩展通配符和变量。这里所谓扩展的意思是程序会把通配符(比如*)替换成合适的文件名,它变量替换成变量值。为了防 止程序作这种替换,您可以使用引号:让我们来看一个例子,假设在当前目录下有一些文件,两个jpg文件, mail.jpg 和tux.jpg。51Testing软件测试网Sn@v5R2b$nM
          编译SHELL脚本
/JY v%g1n,IR5LVc7c K0          #ch#!/bin/sh mod +x filename51Testing软件测试网N l(yRG C:J
 cho *.jpg       ./filename 来执行您的脚本。
"i C1Ed5A:e)R0  这将打印出"mail.jpg tux.jpg"的结果。
;]q/{{;F&t }0    引号 (单引号和双引号) 将防止这种通配符扩展:51Testing软件测试网 Ku-sMTx#Q!e
             #!/bin/sh
eS[h\$z&C0             echo "*.jpg"
U;d+P UU2X ~0             echo '*.jpg'
5}5V vGc0  这将打印"*.jpg" 两次。
||'h LjYW0  单引号更严格一些。它可以防止任何变量扩展。双引号可以防止通配符扩展但允许变量扩展。
9|f'b'gUs5S0         #!/bin/sh
W.e(Iz:eH1y0         echo $SHELL51Testing软件测试网y~YM.S0I
         echo "$SHELL"51Testing软件测试网j&~&J@"aO_0H
         echo '$SHELL'
VZ;M Na5}*C0   运行结果为:
;i#l&g!YF|,S0             /bin/bash
@8m1@!I5l0             /bin/bash
6k@ A7c AII0             $SHELL
D!]IL7VL}l)^-^z\;T0  最后,还有一种防止这种扩展的方法,那就是使用转义字符——反斜杆:51Testing软件测试网V-[-WYj I/~Q"Y\
           echo *.jpg51Testing软件测试网*k+IC_6f5A9m3S$G$j#nK-K
           echo $SHELL
o(o%G@V3` p0  这将输出:
p'za bxvuY.c0           *.jpg
&W+\Iep(c(o4|^e!l0           $SHELL51Testing软件测试网 s[#g!K;ng6EuYM
         6. Here documents
$al E4x!j+Q.Sr0          当要将几行文字传递给一个命令时,here documents(译者注:目前还没有见到过对该词适合的翻译)一种不错的方法。对每个脚本写一段帮助性的文字是很有用的,此时如果我们四有那个 here documents就不必用echo函数一行行输出。 一个 "Here document" 以 << 开头,后面接上一个字符串,这个字符串还必须出现在here document的末尾。下面是一个例子,在该例子中,我们对多个文件进行重命名,并且使用here documents打印帮助:
/rv A4X/s2eBP)vk_0         #!/bin/sh
*j-S1o3b5o6e$th0         # we have less than 3 arguments. Print the help text:
$X"X0yAd0            if [ $# -lt 3 ] ; then51Testing软件测试网 YXV+@L
                 cat <51Testing软件测试网)nLq)P?S5x
                 ren -- renames a number of files using sed regular expressions
#o Lk`&Z0t(R0                 USAGE: ren 'regexp' 'replacement' files...
G q)y6O`x`JQ:n0                 EXAMPLE: rename all *.HTM files in *.html:
-Y2B1M Nq$V0              ren 'HTM$' 'html' *.HTM51Testing软件测试网z{3P(V2kk7z
                 HELP51Testing软件测试网`K6} ^:S%h
              exit 051Testing软件测试网5LY(C2U@,Z(r
            fi
Z:^nJ+xx$_,S0            OLD="$1"
;|!h9_W Dz0            NEW="$2"
~ ]HC5~"h0Sk0          # The shift command removes one argument from the list of
$KEeU^7s0fwO!H#c8B0          # command line arguments.51Testing软件测试网q]/gr)Zx"Z
          shift
}e U7xh&U1S0          shift51Testing软件测试网*s!k o cd6x m
          # $* contains now all the files:51Testing软件测试网A1p\0KAb;q5Mw
         for file in $*; do51Testing软件测试网wo$wb8v)] pX
          if [ -f "$file" ] ; then51Testing软件测试网9wOk9^h%X NqmU
             newfile=`echo "$file" | sed "s/${OLD}/${NEW}/g"`
L|)eeC/~(L!Q0y0             if [ -f "$newfile" ]; then
!k FF:D6q+WuG!~!P0                  echo "ERROR: $newfile exists already"51Testing软件测试网)p;o lj(g*m5@Ssl
             else51Testing软件测试网u,|1pER!W$]
                  echo "renaming $file to $newfile ..."
9T}1S4V3F"q,]2j J*IU0                  mv "$file" "$newfile"51Testing软件测试网)||I(Y*x
              fi51Testing软件测试网3JN-P K/[ ?A"s
            fi
R Z#h @)N:rH0           done51Testing软件测试网 `$A3}i8?iL9R
        这是一个复杂一些的例子。让我们详细讨论一下。第一个if表达式判断输入命令行参数是否小于3个 (特殊变量$# 表示包含参数的个数) 。如果输入参数小于3个,则将帮助文字传递给cat命令,然后由cat命令将其打印在屏幕上。打印帮助文字后程序退出。 如果输入参数等于或大于3个,我们就将第一个参数赋值给变量OLD,第二个参数赋值给变量NEW。下一步,我们使用shift命令将第一个和第二个参数从 参数列表中删除,这样原来的第三个参数就成为参数列表$*的第一个参数。然后我们开始循环,命令行参数列表被一个接一个地被赋值给变量$file。接着我 们判断该文件是否存在,如果存在则通过sed命令搜索和替换来产生新的文件名。然后将反短斜线内命令结果赋值给newfile。这样我们就达到了我们的目 的:得到了旧文件名和新
w/BQ@&V]9?3_a0文件名。然后使用mv命令进行重命名。
uPA1U@0          4)函数51Testing软件测试网"^$c c H$eqS+a'L)eU1V
             如果您写了一些稍微复杂一些的程序,您就会发现在程序中可能在几个地方使用了相同的代码,并且您也会发现,如果我们使用了函数,会方便很多。一个函数是这个样子的:
_nQ `S'J V(e0          functionname()51Testing软件测试网s+Bi-~-h6^,F
          {51Testing软件测试网j"aZ$[O3}T)K;j5~
                 # inside the body $1 is the first argument given to the function51Testing软件测试网${5Q\\Gc'U
                 # $2 the second ...51Testing软件测试网$c @+apjA(z
                             body51Testing软件测试网yLo5mJw#R
            }
5j aE;~B`F0          您需要在每个程序的开始对函数进行声明。
]'fT*il A0   下面是一个叫做xtitlebar的脚本,使用这个脚本您可以改变终端窗口的名称。51Testing软件测试网 \ Q xp1Q;x*w~ ]eR
           这里使用了一个叫做help的函数。正如您可以看到的那样,这个定义的函数被使用了两次。51Testing软件测试网(\\3S4lsO
               #!/bin/sh
3MqC5C/{0]:o8ZN\0               # vim: set sw=4 ts=4 et:51Testing软件测试网4f|K/]2CSfU?
               help()51Testing软件测试网 `X2tO TR
               {
b i)Gy'FzT0              cat <51Testing软件测试网7Jm~|3K;a*onL
                     xtitlebar -- change the name of an xterm, gnome-terminal or kde konsole
:t5Io H}I%N0]t0                     USAGE: xtitlebar [-h] "string_for_titelbar"51Testing软件测试网's8}k!_ o hg
                      OPTIONS: -h help text51Testing软件测试网3tJ/eA~,i/K#M
                     EXAMPLE: xtitlebar "cvs"51Testing软件测试网M7~:nuN;Rq\Um'O
                     HELP
7C9Q q |(}0             exit 0
2\|[V&D;b0               }
|8jd2\@o0               # in case of error or if -h is given we call the function help:
Sf)_[Mz-V0               [ -z "$1" ] && help51Testing软件测试网yEu1s8PN A%E
               [ "$1" = "-h" ] && help51Testing软件测试网Z3| QV2W!{*H
               # send the escape sequence to change the xterm titelbar:
8\#K$X&n.GtL6|1X+t7^0                 echo -e "33]0;$107"51Testing软件测试网qi!r~U eO$f
                #
4j-||2} SaH5MK(v0            在脚本中提供帮助是一种很好的编程习惯,这样方便其他用户(和您)使用和理解脚本。51Testing软件测试网mL0v2p-npp
        命令行参数51Testing软件测试网z/SZ:r4t??7CR/Ct
  我们已经见过$* 和 $1, $2 ... $9 等特殊变量,这些特殊变量包含了用户从命令行输入的参数。迄今为止,我们仅仅了解了一些简单的命令行语法(比如一些强制性的参数和查看帮助的-h选项)。 但是在编写更复杂的程序时,您可能会发现您需要更多的自定义的选项。通常的惯例是在所有可选的参数之前加一个减号,后面再加上参数值 (比如文件名)。有好多方法可以实现对输入参数的分析,但是下面的使用case表达式的例子无遗是一个不错的方法。51Testing软件测试网 \ o r4{d3R6W
         #!/bin/sh
@(F.FJSf0         help()
&_"mOL l4TLz1u"r0         {51Testing软件测试网}P5zC4t4Bh.b
            cat <51Testing软件测试网'Y2y;u w,[.bRa7cV
               This is a generic command line parser demo.
#J&_ R#jE:D Q Y,y0               USAGE EXAMPLE: cmdparser -l hello -f -- -somefile1 somefile251Testing软件测试网r$q;J VHp6?
               HELP51Testing软件测试网I|WR-E @!KB
            exit 051Testing软件测试网;~)|2m _"_6MS
         }
:M&?7\cjKD'L$T3B0          while [ -n "$1" ]; do
#D0X$C ])Q.X@ Q0        case $1 in
+H*c%M&Vh3C0       -h) help;shift 1;; # function help is called
G(o%o N:XM"^+G0       -f) opt_f=1;shift 1;; # variable opt_f is set
M u/p [ }j J0        -l) opt_l=$2;shift 2;; # -l takes an argument -> shift by 2
tI7hO-D,b|Z0        --) shift;break;; # end of options51Testing软件测试网2_ Tes d;We;E
       -*) echo "error: no such option $1. -h for help";exit 1;;
|1C$cJ?T0       *) break;;
.on8L;^"?in0          esac
o'^Z\.u c2q5O0          done51Testing软件测试网EiDE6{ k&p2V
          echo "opt_f is $opt_f"
[ NIwh+d;@{0          echo "opt_l is $opt_l"
$y+y y+VI~0          echo "first arg is $1"
.DaJ[T3y0          echo "2nd arg is $2"
a @)ly-`0  您可以这样运行该脚本:
%b3@+Jv4^5K7{4q5MYo0                 cmdparser -l hello -f -- -somefile1 somefile2
}A)u9Fur0  返回的结果是:51Testing软件测试网3~ p\Qd m
             opt_f is 151Testing软件测试网g3]7xb8v4j Z
             opt_l is hello51Testing软件测试网L2tVk'm9j#}@'E l
             first arg is -somefile1
3kb!H"` F\v$n0             2nd arg is somefile2
3I-V:|HQ0  这个脚本是如何工作的呢?脚本首先在所有输入命令行参数中进行循环,将输入参数与case表达式进行比较,如果匹配则设置一个变量并且移除该参数。根据unix系统的惯例,首先输入的应该是包含减号的参数.
f j]T]f4]N051Testing软件测试网 Ye]5|'cm:rbj
51Testing软件测试网|:HmSA
第2部分 实例51Testing软件测试网~vMN&t6Z

p Te5z[q0    现在我们来讨论编写一个脚本的一般步骤。任何优秀的脚本都应该具有帮助和输入参数。并且写一个伪脚本(framework.sh),该脚本包含了大多数脚本都需要的框架结构,是一个非常不错的主意。这时候,在写一个新的脚本时我们只需要执行一下copy命令:
yK^ VN%}Z5I0cp framework.sh myscript.
&U+uRy?v"R7ngI0 然后再插入自己的函数。
}!Za/g/f3ah0  让我们再看两个例子:51Testing软件测试网5F4y$l3fp3l |y
  二进制到十进制的转换
7]/g{tN5n0  脚本 b2d 将二进制数 (比如 1101) 转换为相应的十进制数。这也是一个用expr命令进行数学运算的例子:
A(|dp{4w0#!/bin/sh51Testing软件测试网6KiYE.EX'b
# vim: set sw=4 ts=4 et:51Testing软件测试网b({pE%EO"JH%F9]
help()51Testing软件测试网9e DQ7A#n`K&\%~4F0|
{51Testing软件测试网6bgDI#\ |
 cat <
@9P rIH$}0b2h -- convert binary to decimal
'u*R:N-_9?0USAGE: b2h [-h] binarynum
(t ~~im7{+W'K)S0OPTIONS: -h help text51Testing软件测试网.|v0MM~3Fn1v)tc
EXAMPLE: b2h 111010
F.A8{v0u h z0ydmq0will return 5851Testing软件测试网]_9pe"L;H&lG z
HELP51Testing软件测试网/r[ZF+CIE
 exit 051Testing软件测试网o1NF Qw8ZRq%x
}
Dv)U3|xqQ0error()
~4|#u8I mV!m{0_0{51Testing软件测试网$no6q {)p&RnU
  # print an error and exit51Testing软件测试网N2GT%LK
  echo "$1"51Testing软件测试网%F*\A$Jc7u6n#W}+pa
  exit 1
l^W2fgUw0}
5z Zd(y QP$e B#VH;M0lastchar()
t%\H]c'?$h0{
va'Q g`5c%t0  # return the last character of a string in $rval51Testing软件测试网*Nx%ys7@;}
  if [ -z "$1" ]; then
}*R^(w I8`y S0    # empty string
[ L+d(q+Wb~%O%Y$|0    rval=""51Testing软件测试网'\XNb;Mz%XZSvt
    return
@;M$M]7F4O0  fi51Testing软件测试网V |E,^benqN
  # wc puts some space behind the output this is why we need sed:51Testing软件测试网(xc Q)p0k"W*O
  numofchar=`echo -n "$1" | wc -c | sed 's/ //g' `51Testing软件测试网C#w&BhjP,X&IcQ
  # now cut out the last char
}.O!q Db? \ P0  rval=`echo -n "$1" | cut -b $numofchar`51Testing软件测试网7O~O |;]~Z e
}
)EA3CQ8b A$PaX0chop()51Testing软件测试网v#`k!Fyf8M@$f-?
{51Testing软件测试网H7S~7g;nT9k {t
  # remove the last character in string and return it in $rval51Testing软件测试网9Evs$y-^Y Tr
  if [ -z "$1" ]; then51Testing软件测试网i ]"^ K1L9k~|1]
    # empty string
|x fd/Te0    rval=""51Testing软件测试网7bE1wu!M?X
    return51Testing软件测试网 B6sn(?/d"[ dWN
  fi
u*u\A"n5OIv ^?0  # wc puts some space behind the output this is why we need sed:
Mt%m'\L E,VA j0  numofchar=`echo -n "$1" | wc -c | sed 's/ //g' `51Testing软件测试网*\0F3jg7]0C*]:qY9UZ?
  if [ "$numofchar" = "1" ]; then
@@3TOF a3e~ M0    # only one char in string
$V_){ Pu)p'Ud0    rval=""51Testing软件测试网;T$| Q,Dw V8F
    return
A7_uENcw0  fi
xt+Cg)`7g \!?0  numofcharminus1=`expr $numofchar "-" 1`51Testing软件测试网@%w`;`?,m
  # now cut all but the last char:
o[u%qv{,Nj Q0  rval=`echo -n "$1" | cut -b 0-${numofcharminus1}`
?_WM'x5A8p+{b%}0}51Testing软件测试网 m5Uch4KQ-w&p9NtM%E
while [ -n "$1" ]; do
`u)K4IA%oT)x9Z)m0case $1 in
(I `X`%z:T0  -h) help;shift 1;; # function help is called
g#@[*j5l0  --) shift;break;; # end of options
jqE.ex'sz?f0  -*) error "error: no such option $1. -h for help";;
^U3q?V#S0L2NoKA0  *) break;;51Testing软件测试网SL#}m e-A1g q
esac51Testing软件测试网$l/BiK3l c#s!p9U
done
fxd1M4c(thP-D0# The main program51Testing软件测试网2OA,zY|2k'CK#v
sum=051Testing软件测试网wXs#Isd|
weight=151Testing软件测试网XZ;hs7kp
# one arg must be given:
DR@ n^0[ -z "$1" ] && help51Testing软件测试网g9GqQ3~+k0d
binnum="$1"
(jQ:er+e.rri3Gb0binnumorig="$1"51Testing软件测试网)\2on3jy4U N
while [ -n "$binnum" ]; do51Testing软件测试网4`"x]}J{C _
  lastchar "$binnum"51Testing软件测试网8kx[H/{"^8I
  if [ "$rval" = "1" ]; then
_m*KGwEy1X0    sum=`expr "$weight" "+" "$sum"`
%DH[`gz5j0  fi51Testing软件测试网8KG X@c7Wy
  # remove the last position in $binnum51Testing软件测试网H3j{6X&s"x!YK
  chop "$binnum"
p:h,s7BkI0J0  binnum="$rval"
+f.STCS9l?0x&k0  weight=`expr "$weight" "*" 2`
V:Uq^$Z'p e?4J8m0done
,] [5uH[&`0echo "binary $binnumorig is decimal $sum"51Testing软件测试网+SYB`f
   该脚本使用的算法是利用十进制和二进制数权值 (1,2,4,8,16,..),比如二进制"10"可以这样转换成十进制:
Lh)d-TN*x00 * 1 + 1 * 2 = 2
4b:iOma6T0  为了得到单个的二进制数我们是用了lastchar 函数。该函数使用wc –c计算字符个数,然后使用cut命令取出末尾一个字符。Chop函数的功能则是移除最后一个字符。
*O9s?,vw*T0    文件循环程序
cGp9jh,[K+mW:Y0  或许您是想将所有发出的邮件保存到一个文件中的人们中的一员,但是在过了几个月以后,这个文件可能会变得很大以至于使对该文件的访问速度变慢。下面的 脚本rotatefile可以解决这个问题。这个脚本可以重命名邮件保存文件(假设为outmail)为outmail.1,而对于outmail.1就 变成了outmail.2 等等等等...51Testing软件测试网V;bG"Na[T
#!/bin/sh51Testing软件测试网 [q9yU3qA{ tKS
# vim: set sw=4 ts=4 et:
II2nA/n*m.sf(U?0ver="0.1"
:}9{S0US0help()
'T'Zr!X)GA0{51Testing软件测试网 T2rR-e)o%k ]:Ja;s
  cat <51Testing软件测试网EM)}0RxJu%a
rotatefile -- rotate the file name
m;oO+kzv S/Dkr0USAGE: rotatefile [-h] filename51Testing软件测试网%A@ ^,V#|C|/^6kN
OPTIONS: -h help text
/M6Z&|&J J0EXAMPLE: rotatefile out51Testing软件测试网S&ou9f S1^ nS(n!\
This will e.g rename out.2 to out.3, out.1 to out.2, out to out.1
4S(Ic"g;U0and create an empty out-file51Testing软件测试网d&O*dD `:g?9w
The max number is 1051Testing软件测试网n TTdN
version $ver
&DUki4y#[(J0HELP
,\*J&U:i/^n3`0  exit 051Testing软件测试网C\)z)EF#`&Ur
}
@:a ahyT}+H/j0error()
XM#U4pU;dq:k:uo0{
u-\i~ J&F5{*byK5u@K0  echo "$1"
;d5hfaMNq)lC0  exit 151Testing软件测试网n5?!X+}p"A'L
}
gm?'_Qj0while [ -n "$1" ]; do51Testing软件测试网 a3}r g|?9P+H
case $1 in51Testing软件测试网Bck5^+SYWX;G
  -h) help;shift 1;;
p3i-iV[0  --) break;;
L,V K#idU0  -*) echo "error: no such option $1. -h for help";exit 1;;51Testing软件测试网l F.Z2jPe
  *) break;;51Testing软件测试网p5]+i+G}$l ZP
esac51Testing软件测试网l~ Z;S4d*f M?
done
Tfb9La5E X!l0# input check:51Testing软件测试网$~:l"N+rKxk&c7Sy
if [ -z "$1" ] ; then51Testing软件测试网 t6Cx!Wi!yf$BL
error "ERROR: you must specify a file, use -h for help"51Testing软件测试网5Q:YM6[!j
fi
W E]SS{0filen="$1"
O{)\ m D*JH8FUEn0# rename any .1 , .2 etc file:51Testing软件测试网'V Zs/?$s{+Y I.K
for n in 9 8 7 6 5 4 3 2 1; do
"r{5s3tL*i0  if [ -f "$filen.$n" ]; then
8nDhKG f/B&K"Q0    p=`expr $n + 1`51Testing软件测试网lx&|q`
    echo "mv $filen.$n $filen.$p"51Testing软件测试网8v2w6u TYc
    mv $filen.$n $filen.$p
'M1H0T7Z$U"H\ ?0  fi51Testing软件测试网oI-}U;n ^}Sl
done51Testing软件测试网Q wA&qS J"i
# rename the original file:51Testing软件测试网+D1m/L!qFM5h?W
if [ -f "$filen" ]; then51Testing软件测试网|x~]:Z'd j
  echo "mv $filen $filen.1"51Testing软件测试网1O2Y~,C8Dx)e
  mv $filen $filen.1
,[WyJ.C Ch.H0fi
5y|K_ Z^0echo touch $filen
)Jx6MI/_ dC0touch $filen51Testing软件测试网znF.J.Q?!|g
  这个脚本是如何工作的呢?在检测用户提供了一个文件名以后,我们进行一个9到1的循环。文件9被命名为10,文件8重命名为9等等。循环完成之后,我们将原始文件命名为文件1同时建立一个与原始文件同名的空文件。
e E-y(CY7hss#{1_$ru0调试
v9Hs'c FfQUNHK(_0  最简单的调试命令当然是使用echo命令。您可以使用echo在任何怀疑出错的地方打印任何变量值。这也是绝大多数的shell程序员要花费80%的时间来调试程序的原因。Shell程序的好处在于不需要重新编译,插入一个echo命令也不需要多少时间。51Testing软件测试网:m#BV9j5m*f
  shell也有一个真实的调试模式。如果在脚本"strangescript" 中有错误,您可以这样来进行调试:51Testing软件测试网7V&l)^0Z-Y?2{db
sh -x strangescript.51Testing软件测试网F#g~v)L9T)k
  这将执行该脚本并显示所有变量的值。51Testing软件测试网8R6a?zJzC5jN d
  shell还有一个不需要执行脚本只是检查语法的模式。可以这样使用:
kh5usj1T0sh -n your_script.51Testing软件测试网6`-E&H-]N:B@)h R
  这将返回所有语法错误。
8d;T)w$|7{Nu0

TAG:

引用 删除 liaomin545   /   2011-09-14 14:25:05
-1
Colorful days 引用 删除 ivwseeqg   /   2011-08-17 09:45:22
5
 

评分:0

我来说两句

Open Toolbar