11shell编程Word文档下载推荐.docx

上传人:b****6 文档编号:21005598 上传时间:2023-01-26 格式:DOCX 页数:40 大小:101.42KB
下载 相关 举报
11shell编程Word文档下载推荐.docx_第1页
第1页 / 共40页
11shell编程Word文档下载推荐.docx_第2页
第2页 / 共40页
11shell编程Word文档下载推荐.docx_第3页
第3页 / 共40页
11shell编程Word文档下载推荐.docx_第4页
第4页 / 共40页
11shell编程Word文档下载推荐.docx_第5页
第5页 / 共40页
点击查看更多>>
下载资源
资源描述

11shell编程Word文档下载推荐.docx

《11shell编程Word文档下载推荐.docx》由会员分享,可在线阅读,更多相关《11shell编程Word文档下载推荐.docx(40页珍藏版)》请在冰豆网上搜索。

11shell编程Word文档下载推荐.docx

itcast:

1000:

itcast,,,:

/home/itcast:

ftp:

115:

125:

ftpdaemon,,,:

/srv/ftp:

用户在命令行输入命令后,一般情况下Shell会fork并exec该命令,但是Shell的内建命令例外,执行内建命令相当于调用Shell进程中的一个函数,并不创建新的进程。

以前学过的cd、alias、umask、exit等命令即是内建命令,凡是用which命令查不到程序文件所在位置的命令都是内建命令,内建命令没有单独的man手册,要在man手册中查看内建命令,应该执行

itcast$manbash-builtins

如export、shift、if、eval、[、for、while等等。

内建命令虽然不创建新的进程,但也会有ExitStatus,通常也用0表示成功非零表示失败,虽然内建命令不创建新的进程,但执行结束后也会有一个状态码,也可以用特殊变量$?

读出。

执行脚本

编写一个简单的脚本test.sh:

#!

/bin/sh

cd..

ls

Shell脚本中用#表示注释,相当于C语言的//注释。

但如果#位于第一行开头,并且是#!

(称为Shebang)则例外,它表示该脚本使用后面指定的解释器/bin/sh解释执行。

如果把这个脚本文件加上可执行权限然后执行:

itcast$chmoda+xtest.sh

itcast$./test.sh

Shell会fork一个子进程并调用exec执行./test.sh这个程序,exec系统调用应该把子进程的代码段替换成./test.sh程序的代码段,并从它的_start开始执行。

然而test.sh是个文本文件,根本没有代码段和_start函数,怎么办呢?

其实exec还有另外一种机制,如果要执行的是一个文本文件,并且第一行用Shebang指定了解释器,则用解释器程序的代码段替换当前进程,并且从解释器的_start开始执行,而这个文本文件被当作命令行参数传给解释器。

因此,执行上述脚本相当于执行程序

itcast$/bin/sh./test.sh

以这种方式执行不需要test.sh文件具有可执行权限。

如果将命令行下输入的命令用()括号括起来,那么也会fork出一个子Shell执行小括号中的命令,一行中可以输入由分号;

隔开的多个命令,比如:

itcast$(cd..;

ls-l)

和上面两种方法执行Shell脚本的效果是相同的,cd..命令改变的是子Shell的PWD,而不会影响到交互式Shell。

然而命令

itcast$cd..;

ls-l

则有不同的效果,cd..命令是直接在交互式Shell下执行的,改变交互式Shell的PWD,然而这种方式相当于这样执行Shell脚本:

itcast$source./test.sh

或者

itcast$../test.sh

source或者.命令是Shell的内建命令,这种方式也不会创建子Shell,而是直接在交互式Shell下逐行执行脚本中的命令。

基本语法

变量

按照惯例,Shell变量通常由字母加下划线开头,由任意长度的字母、数字、下划线组成。

有两种类型的Shell变量:

1.环境变量

环境变量可以从父进程传给子进程,因此Shell进程的环境变量可以从当前Shell进程传给fork出来的子进程。

用printenv命令可以显示当前Shell进程的环境变量。

2.本地变量

只存在于当前Shell进程,用set命令可以显示当前Shell进程中定义的所有变量(包括本地变量和环境变量)和函数。

环境变量是任何进程都有的概念,而本地变量是Shell特有的概念。

在Shell中,环境变量和本地变量的定义和用法相似。

在Shell中定义或赋值一个变量:

itcast$VARNAME=value

注意等号两边都不能有空格,否则会被Shell解释成命令和命令行参数。

一个变量定义后仅存在于当前Shell进程,它是本地变量,用export命令可以把本地变量导出为环境变量,定义和导出环境变量通常可以一步完成:

itcast$exportVARNAME=value

也可以分两步完成:

itcast$exportVARNAME

用unset命令可以删除已定义的环境变量或本地变量。

itcast$unsetVARNAME

如果一个变量叫做VARNAME,用'

VARNAME'

可以表示它的值,在不引起歧义的情况下也可以用VARNAME表示它的值。

通过以下例子比较这两种表示法的不同:

itcast$echo$SHELL

注意,在定义变量时不用“'

”取变量值时要用。

和C语言不同的是,Shell变量不需要明确定义类型,事实上Shell变量的值都是字符串,比如我们定义VAR=45,其实VAR的值是字符串45而非整数。

Shell变量不需要先定义后使用,如果对一个没有定义的变量取值,则值为空字符串。

文件名代换(Globbing)

这些用于匹配的字符称为通配符(Wildcard),如:

*?

[]具体如下:

*匹配0个或多个任意字符

?

匹配一个任意字符

[若干字符]匹配方括号中任意一个字符的一次出现

itcast$ls/dev/ttyS*

itcast$lsch0?

.doc

itcast$lsch0[0-2].doc

itcast$lsch[012][0-9].doc

注意,Globbing所匹配的文件名是由Shell展开的,也就是说在参数还没传给程序之前已经展开了,比如上述lsch0[012].doc命令,如果当前目录下有ch00.doc和ch02.doc,则传给ls命令的参数实际上是这两个文件名,而不是一个匹配字符串。

命令代换

由“`”反引号括起来的也是一条命令,Shell先执行该命令,然后将输出结果立刻代换到当前命令行中。

例如定义一个变量存放date命令的输出:

itcast$DATE=`date`

itcast$echo$DATE

命令代换也可以用$()表示:

itcast$DATE=$(date)

算术代换

使用$(()),用于算术计算,(())中的Shell变量取值将转换成整数,同样含义的$[]等价例如:

itcast$VAR=45

itcast$echo$(($VAR+3))等价于echo$[VAR+3]或$[$VAR+3]

$(())中只能用+-*/和()运算符,并且只能做整数运算。

$[base#n],其中base表示进制,n按照base进制解释,后面再有运算数,按十进制解释。

echo$[2#10+11]

echo$[8#10+11]

echo$[16#10+11]

转义字符

和C语言类似,\在Shell中被用作转义字符,用于去除紧跟其后的单个字符的特殊意义(回车除外),换句话说,紧跟其后的字符取字面值。

例如:

itcast$echo\$SHELL

$SHELL

itcast$echo\\

\

比如创建一个文件名为“$$”的文件($间含有空格)可以这样:

itcast$touch\$\\$

还有一个字符虽然不具有特殊含义,但是要用它做文件名也很麻烦,就是-号。

如果要创建一个文件名以-号开头的文件,这样是不正确的:

itcast$touch-hello

touch:

invalidoption--h

Try`touch--help'

formoreinformation.

即使加上\转义也还是报错:

itcast$touch\-hello

因为各种UNIX命令都把-号开头的命令行参数当作命令的选项,而不会当作文件名。

如果非要处理以-号开头的文件名,可以有两种办法:

itcast$touch./-hello

itcast$touch---hello

\还有一种用法,在\后敲回车表示续行,Shell并不会立刻执行命令,而是把光标移到下一行,给出一个续行提示符>

,等待用户继续输入,最后把所有的续行接到一起当作一个命令执行。

itcast$ls\

>

-l

(ls-l命令的输出)

单引号

和C语言同,Shell脚本中的单引号和双引号一样都是字符串的界定符(双引号下一节介绍),而不是字符的界定符。

单引号用于保持引号内所有字符的字面值,即使引号内的\和回车也不例外,但是字符串中不能出现单引号。

如果引号没有配对就输入回车,Shell会给出续行提示符,要求用户把引号配上对。

itcast$echo'

$SHELL'

ABC\(回车)

DE'

(再按一次回车结束命令)

ABC\

DE

双引号

被双引号用括住的内容,将被视为单一字串。

它防止通配符扩展,但允许变量扩展。

这点与单引号的处理方式不同

itcast$echo"

$DATE"

$DATE'

再比如:

itcast$VAR=200

itcast$echo$VAR

200

$VAR'

$VAR

$VAR"

Shell脚本语法

条件测试

命令test或[可以测试一个条件是否成立,如果测试结果为真,则该命令的ExitStatus为0,如果测试结果为假,则命令的ExitStatus为1(注意与C语言的逻辑表示正好相反)。

例如测试两个数的大小关系:

itcast@ubuntu:

~$var=2

~$test$var-gt1

~$echo$?

~$test$var-gt3

1

~$[$var-gt3]

~$

虽然看起来很奇怪,但左方括号[确实是一个命令的名字,传给命令的各参数之间应该用空格隔开,比如:

$VAR、-gt、3、]是[命令的四个参数,它们之间必须用空格隔开。

命令test或[的参数形式是相同的,只不过test命令不需要]参数。

以[命令为例,常见的测试命令如下表所示:

[-dDIR]如果DIR存在并且是一个目录则为真

[-fFILE]如果FILE存在且是一个普通文件则为真

[-zSTRING]如果STRING的长度为零则为真

[-nSTRING]如果STRING的长度非零则为真

[STRING1=STRING2]如果两个字符串相同则为真

[STRING1!

=STRING2]如果字符串不相同则为真

[ARG1OPARG2]ARG1和ARG2应该是整数或者取值为整数的变量,OP是-eq(等于)-ne(不等于)-lt(小于)-le(小于等于)-gt(大于)-ge(大于等于)之中的一个

和C语言类似,测试条件之间还可以做与、或、非逻辑运算:

[!

EXPR]EXPR可以是上表中的任意一种测试条件,!

表示“逻辑反(非)”

[EXPR1-aEXPR2]EXPR1和EXPR2可以是上表中的任意一种测试条件,-a表示“逻辑与”

[EXPR1-oEXPR2]EXPR1和EXPR2可以是上表中的任意一种测试条件,-o表示“逻辑或”

$VAR=abc

$[-dDesktop-a$VAR='

abc'

]

$echo$?

注意,如果上例中的$VAR变量事先没有定义,则被Shell展开为空字符串,会造成测试条件的语法错误(展开为[-dDesktop-a=‘abc’]),作为一种好的Shell编程习惯,应该总是把变量取值放在双引号之中(展开为[-dDesktop-a“”=‘abc’]):

$unsetVAR

bash:

[:

toomanyarguments

$[-dDesktop-a"

='

分支

if/then/elif/else/fi

和C语言类似,在Shell中用if、then、elif、else、fi这几条命令实现分支控制。

这种流程控制语句本质上也是由若干条Shell命令组成的,例如先前讲过的

if[-f~/.bashrc];

then

.~/.bashrc

fi

其实是三条命令,if[-f∼/.bashrc]是第一条,then.∼/.bashrc是第二条,fi是第三条。

如果两条命令写在同一行则需要用;

号隔开,一行只写一条命令就不需要写;

号了,另外,then后面有换行,但这条命令没写完,Shell会自动续行,把下一行接在then后面当作一条命令处理。

和[命令一样,要注意命令和各参数之间必须用空格隔开。

if命令的参数组成一条子命令,如果该子命令的ExitStatus为0(表示真),则执行then后面的子命令,如果ExitStatus非0(表示假),则执行elif、else或者fi后面的子命令。

if后面的子命令通常是测试命令,但也可以是其它命令。

Shell脚本没有{}括号,所以用fi表示if语句块的结束。

见下例:

if[-f/bin/bash]

then

echo"

/bin/bashisafile"

else

/bin/bashisNOTafile"

if:

;

thenecho"

alwaystrue"

fi

“:

”是一个特殊的命令,称为空命令,该命令不做任何事,但ExitStatus总是真。

此外,也可以执行/bin/true或/bin/false得到真或假的ExitStatus。

再看一个例子:

echo"

Isitmorning?

Pleaseansweryesorno."

readYES_OR_NO

if["

$YES_OR_NO"

="

yes"

];

Goodmorning!

"

elif["

no"

Goodafternoon!

else

Sorry,$YES_OR_NOnotrecognized.Enteryesorno."

exit1

exit0

上例中的read命令的作用是等待用户输入一行字符串,将该字符串存到一个Shell变量中。

此外,Shell还提供了&

&

和||语法,和C语言类似,具有Short-circuit特性,很多Shell脚本喜欢写成这样:

test"

$(whoami)"

!

='

root'

&

(echoyouareusinganon-privilegedaccount;

exit1)

相当于“if…then…”,而||相当于“ifnot…then…”。

和||用于连接两个命令,而上面讲的-a和-o仅用于在测试表达式中连接两个测试条件,要注意它们的区别,例如:

-gt1-a"

-lt3

和以下写法是等价的

-gt1&

test"

case/esac

case命令可类比C语言的switch/case语句,esac表示case语句块的结束。

C语言的case只能匹配整型或字符型常量表达式,而Shell脚本的case可以匹配字符串和Wildcard,每个匹配分支可以有若干条命令,末尾必须以;

结束,执行时找到第一个匹配的分支并执行相应的命令,然后直接跳到esac之后,不需要像C语言一样用break跳出。

case"

in

yes|y|Yes|YES)

GoodMorning!

[nN]*)

GoodAfternoon!

*)

exit1;

esac

使用case语句的例子可以在系统服务的脚本目录/etc/init.d中找到。

这个目录下的脚本大多具有这种形式(以/etc/init.d/nfs-kernel-server为例):

$1"

start)

...

;

stop)

reload|force-reload)

restart)

*)

log_success_msg"

Usage:

nfs-kernel-server{start|stop|status|reload|force-reload|restart}"

启动nfs-kernel-server服务的命令是

$sudo/etc/init.d/nfs-kernel-serverstart

$1是一个特殊变量,在执行脚本时自动取值为第一个命令行参数,也就是start,所以进入start)分支执行相关的命令。

同理,命令行参数指定为stop、reload或restart可以进入其它分支执行停止服务、重新加载配置文件或重新启动服务的相关命令。

循环

for/do/done

Shell脚本的for循环结构和C语言很不一样,它类似于某些编程语言的foreach循环。

forFRUITinapplebananapear;

do

Ilike$FRUIT"

done

FRUIT是一个循环变量,第一次循环$FRUIT的取值是apple,第二次取值是banana,第三次取值是pear。

再比如,要将当前目录下的chap0、chap1、chap2等文件名改为chap0~、chap1~、chap2~等(按惯例,末尾有~字符的文件名表示临时文件),这个命令可以这样写:

$forFILENAMEinchap?

domv$FILENAME$FILENAME~;

done

也可以这样写:

$forFILENAMEin`lschap?

`;

while/do/done

while的用法和C语言类似。

比如一个验证密码的脚本:

Enterpassword:

readTRY

while["

$TRY"

="

secret"

Sorry,tryagain"

readTRY

下面的例子通过算术运算控制循环的次数:

COUNTER=1

$COUNTER"

-lt10];

Herewegoagain"

COUNTER=$(($COUNTER+1))

另,Shell还有until循环,类似C语言的do…while。

如有兴趣可在课后自行扩展学习。

break和continue

break[n]可以指定跳出几层循环;

continue跳过本次循环,但不会跳出循环。

即break跳出,continue跳过。

练习:

将上面验证密码的程序修改一下,如果用户输错五次密码就报错退出。

位置参数和特殊变量

有很多特殊变量是被Shell自动赋值的,我们已经遇到了$?

和$1。

其他常用的位置参数和特殊变量在这里总结一下:

$0相当于C语言main函数的argv[0]

$1、$2...这些称为位置参数(PositionalParameter),相当于C语言main函数的argv[1]、argv[2]...

$#相当于C语言main函数的argc-1,注意这里的

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

当前位置:首页 > 高等教育 > 其它

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1