shell编程.docx

上传人:b****2 文档编号:2590670 上传时间:2023-05-04 格式:DOCX 页数:20 大小:23.33KB
下载 相关 举报
shell编程.docx_第1页
第1页 / 共20页
shell编程.docx_第2页
第2页 / 共20页
shell编程.docx_第3页
第3页 / 共20页
shell编程.docx_第4页
第4页 / 共20页
shell编程.docx_第5页
第5页 / 共20页
shell编程.docx_第6页
第6页 / 共20页
shell编程.docx_第7页
第7页 / 共20页
shell编程.docx_第8页
第8页 / 共20页
shell编程.docx_第9页
第9页 / 共20页
shell编程.docx_第10页
第10页 / 共20页
shell编程.docx_第11页
第11页 / 共20页
shell编程.docx_第12页
第12页 / 共20页
shell编程.docx_第13页
第13页 / 共20页
shell编程.docx_第14页
第14页 / 共20页
shell编程.docx_第15页
第15页 / 共20页
shell编程.docx_第16页
第16页 / 共20页
shell编程.docx_第17页
第17页 / 共20页
shell编程.docx_第18页
第18页 / 共20页
shell编程.docx_第19页
第19页 / 共20页
shell编程.docx_第20页
第20页 / 共20页
亲,该文档总共20页,全部预览完了,如果喜欢就下载吧!
下载资源
资源描述

shell编程.docx

《shell编程.docx》由会员分享,可在线阅读,更多相关《shell编程.docx(20页珍藏版)》请在冰点文库上搜索。

shell编程.docx

shell编程

Shell编程基础

我们可以使用任意一种文字编辑器,比如gedit、kedit、emacs、vi等来编写shell脚本,它必须以如下行开始(必须放在文件的第一行):

# !

/bin/sh

符号#!

用来告诉系统执行该脚本的程序,本例使用/bin/sh。

编辑结束并保存后,如果要执行该脚本,必须先使其可执行:

chmod+xfilename

此后在该脚本所在目录下,输入./filename即可执行该脚本。

变量赋值和引用

Shell编程中,使用变量无需事先声明,同时变量名的命名须遵循如下规则:

1.首个字符必须为字母(a-z,A-Z)

2.中间不能有空格,可以使用下划线(_)

3.不能使用标点符号

4.不能使用bash里的关键字(可用help命令查看保留关键字)

需要给变量赋值时,可以这么写:

变量名=值

要取用一个变量的值,只需在变量名前面加一个$(注意:

给变量赋值的时候,不能在"="两边留空格)

#!

/bin/sh

#对变量赋值:

a="helloworld"#等号两边均不能有空格存在

#打印变量a的值:

echo"Ais:

"$a

挑个自己喜欢的编辑器,输入上述内容,并保存为文件first,然后执行chmod+xfirst使其可执行,最后输入./first执行该脚本。

其输出结果如下:

 

Ais:

helloworld

有时候变量名可能会和其它文字混淆,比如:

num=2

echo"thisisthe$numnd"

上述脚本并不会输出"thisisthe2nd"而是"thisisthe";这是由于shell会去搜索变量numnd的值,而实际上这个变量此时并没有值。

这时,我们可以用花括号来告诉shell要打印的是num变量:

num=2

echo"thisisthe${num}nd"

其输出结果为:

thisisthe2nd

Shell脚本中有许多变量是系统自动设定的,我们将在用到这些变量时再作说明。

除了只在脚本内有效的普通shell变量外,还有环境变量,即那些由export关键字处理过的变量。

本文不讨论环境变量,因为它们一般只在登录脚本中用到。

Shell里的流程控制

if语句

"if"表达式如果条件为真,则执行then后的部分:

if....;then

....

elif....;then

....

else

....

fi

大多数情况下,可以使用测试命令来对条件进行测试,比如可以比较字符串、判断文件是否存在及是否可读等等……通常用"[]"来表示条件测试,注意这里的空格很重要,要确保方括号前后的空格。

[-f"somefile"]:

判断是否是一个文件

[-x"/bin/ls"]:

判断/bin/ls是否存在并有可执行权限

[-n"$var"]:

判断$var变量是否有值

["$a"="$b"]:

判断$a和$b是否相等

执行mantest可以查看所有测试表达式可以比较和判断的类型。

下面是一个简单的if语句:

#!

/bin/sh

if[${SHELL}="/bin/bash"];then

echo"yourloginshellisthebash(bourneagainshell)"

else

echo"yourloginshellisnotbashbut${SHELL}"

fi

变量$SHELL包含有登录shell的名称,我们拿它和/bin/bash进行比较以判断当前使用的shell是否为bash。

&&和||操作符

熟悉C语言的朋友可能会喜欢下面的表达式:

[-f"/etc/shadow"]&&echo"Thiscomputerusesshadowpasswords"

这里的&&就是一个快捷操作符,如果左边的表达式为真则执行右边的语句,你也可以把它看作逻辑运算里的与操作。

上述脚本表示如果/etc/shadow文件存在,则打印”Thiscomputerusesshadowpasswords”。

同样shell编程中还可以用或操作(||),例如:

#!

/bin/sh

mailfolder=/var/spool/mail/james

[-r"$mailfolder"]||{echo"Cannotread$mailfolder" ;exit1;}

echo"$mailfolderhasmailfrom:

"

grep"^From"$mailfolder

该脚本首先判断mailfolder是否可读,如果可读则打印该文件中的"From"一行。

如果不可读则或操作生效,打印错误信息后脚本退出。

需要注意的是,这里我们必须使用如下两个命令:

-打印错误信息

-退出程序

我们使用花括号以匿名函数的形式将两个命令放到一起作为一个命令使用;普通函数稍后再作说明。

即使不用与和或操作符,我们也可以用if表达式完成任何事情,但是使用与或操作符会更便利很多。

case语句

case表达式可以用来匹配一个给定的字符串,而不是数字(可别和C语言里的switch...case混淆)。

case...in

...)dosomethinghere ;;

esac

让我们看一个例子,file命令可以辨别出一个给定文件的文件类型,如:

filelf.gz,其输出结果为:

lf.gz:

gzipcompresseddata,deflated,originalfilename,

lastmodified:

MonAug2723:

09:

182001,os:

Unix

我们利用这点写了一个名为smartzip的脚本,该脚本可以自动解压bzip2,gzip和zip类型的压缩文件:

#!

/bin/sh

ftype=`file"$1"`#Note'and`isdifferent

case"$ftype"in

"$1:

Ziparchive"*)

unzip"$1" ;;

"$1:

gzipcompressed"*)

gunzip"$1" ;;

"$1:

bzip2compressed"*)

bunzip2"$1" ;;

*)echo"File$1cannotbeuncompressedwithsmartzip";;

esac

你可能注意到上面使用了一个特殊变量$1,该变量包含有传递给该脚本的第一个参数值。

也就是说,当我们运行:

smartziparticles.zip

$1就是字符串articles.zip。

select语句

select表达式是bash的一种扩展应用,擅长于交互式场合。

用户可以从一组不同的值中进行选择:

selectvarin... ;do

 break;

done

....now$varcanbeused....

下面是一个简单的示例:

#!

/bin/sh

echo"WhatisyourfavouriteOS?

"

selectvarin"Linux""GnuHurd""FreeBSD""Other";do

    break;

done

echo"Youhaveselected$var"

1.如果以上脚本运行出现select:

NOTFOUND将#!

/bin/sh改为#!

/bin/bash找了半天才找到的答案

该脚本的运行结果如下:

WhatisyourfavouriteOS?

1)Linux

2)GnuHurd

3)FreeBSD

4)Other

#?

1

YouhaveselectedLinux

while/for循环

在shell中,可以使用如下循环:

while...;do

....

done

只要测试表达式条件为真,则while循环将一直运行。

关键字"break"用来跳出循环,而关键字”continue”则可以跳过一个循环的余下部分,直接跳到下一次循环中。

for循环会查看一个字符串行表(字符串用空格分隔),并将其赋给一个变量:

forvarin....;do

....

done

下面的示例会把ABC分别打印到屏幕上:

#!

/bin/sh

forvarinABC ;do

echo"varis$var"

done

下面是一个实用的脚本showrpm,其功能是打印一些RPM包的统计信息:

#!

/bin/sh

#listacontentsummaryofanumberofRPMpackages

#USAGE:

showrpmrpmfile1rpmfile2...

#EXAMPLE:

showrpm/cdrom/RedHat/RPMS/*.rpm

forrpmpackagein$*;do

if[-r"$rpmpackage"];then

echo"===============$rpmpackage=============="

rpm-qi-p$rpmpackage

else

echo"ERROR:

cannotreadfile$rpmpackage"

fi

done

这里出现了第二个特殊变量$*,该变量包含有输入的所有命令行参数值。

如果你运行showrpmopenssh.rpmw3m.rpmwebgrep.rpm,那么$*就包含有3个字符串,即openssh.rpm,w3m.rpm和webgrep.rpm。

Shell里的一些特殊符号

引号

在向程序传递任何参数之前,程序会扩展通配符和变量。

这里所谓的扩展是指程序会把通配符(比如*)替换成适当的文件名,把变量替换成变量值。

我们可以使用引号来防止这种扩展,先来看一个例子,假设在当前目录下有两个jpg文件:

mail.jpg和tux.jpg。

#!

/bin/sh

echo*.jpg

运行结果为:

mail.jpgtux.jpg

引号(单引号和双引号)可以防止通配符*的扩展:

#!

/bin/sh

echo"*.jpg"

echo'*.jpg'

其运行结果为:

*.jpg

*.jpg

其中单引号更严格一些,它可以防止任何变量扩展;而双引号可以防止通配符扩展但允许变量扩展:

#!

/bin/sh

echo$SHELL

echo"$SHELL"

echo'$SHELL'

运行结果为:

/bin/bash

/bin/bash

$SHELL

此外还有一种防止这种扩展的方法,即使用转义字符——反斜杆:

\:

echo\*.jpg

echo\$SHELL

输出结果为:

*.jpg

$SHELL

HereDocument

当要将几行文字传递给一个命令时,用heredocuments是一种不错的方法。

对每个脚本写一段帮助性的文字是很有用的,此时如果使用heredocuments就不必用echo函数一行行输出。

Heredocument以<<开头,后面接上一个字符串,这个字符串还必须出现在heredocument的末尾。

下面是一个例子,在该例子中,我们对多个文件进行重命名,并且使用heredocuments打印帮助:

#!

/bin/sh

#wehavelessthan3arguments.Printthehelptext:

if[$#-lt3] ;then

cat<

ren--renamesanumberoffilesusingsedregularexpressionsUSAGE:

ren'regexp''replacement'files...

EXAMPLE:

renameall*.HTMfilesin*.html:

ren'HTM$''html'*.HTM

HELP

exit0

fi

OLD="$1"

NEW="$2"

#Theshiftcommandremovesoneargumentfromthelistof

#commandlinearguments.

shift

shift

#$*containsnowallthefiles:

forfilein$*;do

if[-f"$file"] ;then

newfile=`echo"$file"|sed"s/${OLD}/${NEW}/g"`

if[-f"$newfile"];then

    echo"ERROR:

$newfileexistsalready"

else

echo"renaming$fileto$newfile..."

mv"$file""$newfile"

fi

fi

done

这个示例有点复杂,我们需要多花点时间来说明一番。

第一个if表达式判断输入命令行参数是否小于3个(特殊变量$#表示包含参数的个数)。

如果输入参数小于3个,则将帮助文字传递给cat命令,然后由cat命令将其打印在屏幕上。

打印帮助文字后程序退出。

如果输入参数等于或大于3个,我们就将第一个参数赋值给变量OLD,第二个参数赋值给变量NEW。

下一步,我们使用shift命令将第一个和第二个参数从参数列表中删除,这样原来的第三个参数就成为参数列表$*的第一个参数。

然后我们开始循环,命令行参数列表被一个接一个地被赋值给变量$file。

接着我们判断该文件是否存在,如果存在则通过sed命令搜索和替换来产生新的文件名。

然后将反短斜线内命令结果赋值给newfile。

这样我们就达到了目的:

得到了旧文件名和新文件名。

然后使用mv命令进行重命名

Shell里的函数

如果你写过比较复杂的脚本,就会发现可能在几个地方使用了相同的代码,这时如果用上函数,会方便很多。

函数的大致样子如下:

functionname()

{

#insidethebody$1isthefirstargumentgiventothefunction

#$2thesecond...

body

}

你需要在每个脚本的开始对函数进行声明。

下面是一个名为xtitlebar的脚本,它可以改变终端窗口的名称。

这里使用了一个名为help的函数,该函数在脚本中使用了两次:

#!

/bin/sh

#vim:

setsw=4ts=4et:

help()

{

cat<

xtitlebar--changethenameofanxterm,gnome-terminalorkdekonsole

USAGE:

xtitlebar[-h]"string_for_titelbar"

OPTIONS:

-hhelptext

EXAMPLE:

xtitlebar"cvs"

HELP

exit0

}

#incaseoferrororif-hisgivenwecallthefunctionhelp:

[-z"$1"]&&help

["$1"="-h"]&&help

#sendtheescapesequencetochangethextermtitelbar:

echo-e"33]0;$107"

#

在脚本中提供帮助是一种很好的编程习惯,可以方便其他用户(和自己)使用和理解脚本。

命令行参数

我们已经见过$*和$1,$2...$9等特殊变量,这些特殊变量包含了用户从命令行输入的参数。

迄今为止,我们仅仅了解了一些简单的命令行语法(比如一些强制性的参数和查看帮助的-h选项)。

但是在编写更复杂的程序时,您可能会发现您需要更多的自定义的选项。

通常的惯例是在所有可选的参数之前加一个减号,后面再加上参数值(比如文件名)。

有好多方法可以实现对输入参数的分析,但是下面的使用case表达式的例子无疑是一个不错的方法。

#!

/bin/sh

help()

{

cat<

Thisisagenericcommandlineparserdemo.

USAGEEXAMPLE:

cmdparser-lhello-f---somefile1somefile2

HELP

exit0

}

while[-n"$1"];do

case$1in

-h)help;shift1;;#functionhelpiscalled

-f)opt_f=1;shift1;;#variableopt_fisset

-l)opt_l=$2;shift2;;#-ltakesanargument->shiftby2

--)shift;break;;#endofoptions

-*)echo"error:

nosuchoption$1.-hforhelp";exit1;;

*)break;;

esac

done

echo"opt_fis$opt_f"

echo"opt_lis$opt_l"

echo"firstargis$1"

echo"2ndargis$2"

你可以这样运行该脚本:

cmdparser-lhello-f---somefile1somefile2

返回结果如下:

opt_fis1

opt_lishello

firstargis-somefile1

2ndargissomefile2

这个脚本是如何工作的呢?

脚本首先在所有输入命令行参数中进行循环,将输入参数与case表达式进行比较,如果匹配则设置一个变量并且移除该参数。

根据unix系统的惯例,首先输入的应该是包含减号的参数。

Shell脚本示例

一般编程步骤

现在我们来讨论编写一个脚本的一般步骤。

任何优秀的脚本都应该具有帮助和输入参数。

写一个框架脚本(framework.sh),该脚本包含了大多数脚本需要的框架结构,是一个非常不错的主意。

这样一来,当我们开始编写新脚本时,可以先执行如下命令:

cpframework.shmyscript

然后再插入自己的函数。

让我们来看看如下两个示例。

二进制到十进制的转换

脚本b2d将二进制数(比如1101)转换为相应的十进制数。

这也是一个用expr命令进行数学运算的例子:

#!

/bin/sh

#vim:

setsw=4ts=4et:

help()

{

cat<

b2d--convertbinarytodecimal

USAGE:

b2d[-h]binarynum

OPTIONS:

-hhelptext

EXAMPLE:

b2d111010

willreturn58

HELP

exit0

}

error()

{

#printanerrorandexit

echo"$1"

exit1

}

lastchar()

{

#returnthelastcharacterofastringin$rval

if[-z"$1"];then

#emptystring

rval=""

return

fi

#wcputssomespacebehindtheoutputthisiswhyweneedsed:

numofchar=`echo-n"$1"|wc-c|sed's///g'`

#nowcutoutthelastchar

rval=`echo-n"$1"|cut-b$numofchar`

}

chop()

{

#removethelastcharacterinstringandreturnitin$rval

if[-z"$1"];then

#emptystring

rval=""

return

fi

#wcputssomespacebehindtheoutputthisiswhyweneedsed:

numofchar=`echo-n"$1"|wc-c|sed's///g'`

if["$numofchar"="1"];then

#onlyonecharinstring

rval=""

return

fi

numofcharminus1=`expr$numofchar"-"1`

#nowcutallbutthelastchar:

rval=`echo-n"$1"|cut-b-$numofcharminus1`

#原来的rval=`echo-n"$1"|cut-b0-${numofcharminus1}`运行时出错.

#原因是cut从1开始计数,应该是cut-b1-${numofcharminus1}

}

while[-n"$1"];do

case$1in

-h)help;shift1;;#functionhelpiscalled

--)shift;break;;#endofoptions

-*)error"error:

nosuchoption$1.-hforhelp";;

*)break;;

esac

done

#Themainprogram

sum=0

weight=1

#oneargmustbegiven:

[-z"$1"]&&help

binnum="$1"

binnumorig="$1"

while[-n"$binnum"];do

lastchar"$binnum"

if["$rval"="1"];then

sum=`expr"$weight""+""$sum"`

fi

#removethelastpositionin$binnum

chop"$binnum"

binnum="$rval"

weight=`expr"$weight""*"2`

done

echo"binary$binnumorigisdecimal$sum"

#

该脚本使用的算法是利用十进制和二进制数权值(1,2,4,8,16,..),比如二进制"10"可以这样转

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 解决方案 > 学习计划

copyright@ 2008-2023 冰点文库 网站版权所有

经营许可证编号:鄂ICP备19020893号-2