Shell 脚本知识回顾 (六) —— Shell 函数

来源:互联网 发布:html5 大数据动态展示 编辑:程序博客网 时间:2024/04/29 14:28

一、Shell函数:Shell函数返回值、删除函数、在终端调用函数

函数可以让我们将一个复杂功能划分成若干模块,让程序结构更加清晰,代码重复利用率更高。像其他编程语言一样,Shell 也支持函数。Shell 函数必须先定义后使用。


Shell 函数的定义格式如下:
function_name () {    list of commands    [ return value ]}
如果你愿意,也可以在函数名前加上关键字 function:
function function_name () {    list of commands    [ return value ]}
函数返回值,可以显式增加return语句;如果不加,会将最后一条命令运行结果作为返回值。

Shell 函数返回值只能是整数,一般用来表示函数执行成功与否,0表示成功,其他值表示失败。如果 return 其他数据,比如一个字符串,往往会得到错误提示:“numeric argument required”。

如果一定要让函数返回字符串,那么可以先定义一个变量,用来接收函数的计算结果,脚本在需要的时候访问这个变量来获得函数返回值。

先来看一个例子:
[cpp] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. #!/bin/bash  
  2. # Define your function here  
  3. Hello () {  
  4.    echo "Url is http://see.xidian.edu.cn/cpp/shell/"  
  5. }  
  6. # Invoke your function  
  7. Hello  
运行结果:
$./test.shHello World$
调用函数只需要给出函数名,不需要加括号。

再来看一个带有return语句的函数:
[cpp] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. #!/bin/bash  
  2. funWithReturn(){  
  3.     echo "The function is to get the sum of two numbers..."  
  4.     echo -n "Input first number: "  
  5.     read aNum  
  6.     echo -n "Input another number: "  
  7.     read anotherNum  
  8.     echo "The two numbers are $aNum and $anotherNum !"  
  9.     return $(($aNum+$anotherNum))  
  10. }  
  11. funWithReturn  
  12. # Capture value returnd by last command  
  13. ret=$?  
  14. echo "The sum of two numbers is $ret !"  
运行结果:
The function is to get the sum of two numbers...Input first number: 25Input another number: 50The two numbers are 25 and 50 !The sum of two numbers is 75 !
函数返回值在调用该函数后通过 $? 来获得。

再来看一个函数嵌套的例子:
[cpp] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. #!/bin/bash  
  2. # Calling one function from another  
  3. number_one () {  
  4.    echo "Url_1 is http://see.xidian.edu.cn/cpp/shell/"  
  5.    number_two  
  6. }  
  7. number_two () {  
  8.    echo "Url_2 is http://see.xidian.edu.cn/cpp/u/xitong/"  
  9. }  
  10. number_one  
运行结果
Url_1 is http://see.xidian.edu.cn/cpp/shell/Url_2 is http://see.xidian.edu.cn/cpp/u/xitong/

像删除变量一样,删除函数也可以使用 unset 命令,不过要加上 .f 选项,如下所示:

[cpp] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. $unset .f function_name  

如果你希望直接从终端调用函数,可以将函数定义在主目录下的 .profile 文件,这样每次登录后,在命令提示符后面输入函数名字就可以立即调用。


二、Shell函数参数

在Shell中,调用函数时可以向其传递参数。在函数体内部,通过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数...

带参数的函数示例:

[cpp] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. #!/bin/bash  
  2. funWithParam(){  
  3.     echo "The value of the first parameter is $1 !"  
  4.     echo "The value of the second parameter is $2 !"  
  5.     echo "The value of the tenth parameter is $10 !"  
  6.     echo "The value of the tenth parameter is ${10} !"  
  7.     echo "The value of the eleventh parameter is ${11} !"  
  8.     echo "The amount of the parameters is $# !"  # 参数个数  
  9.     echo "The string of the parameters is $* !"  # 传递给函数的所有参数  
  10. }  
  11. funWithParam 1 2 3 4 5 6 7 8 9 34 73  
运行脚本:

The value of the first parameter is 1 !The value of the second parameter is 2 !The value of the tenth parameter is 10 !The value of the tenth parameter is 34 !The value of the eleventh parameter is 73 !The amount of the parameters is 12 !The string of the parameters is 1 2 3 4 5 6 7 8 9 34 73 !"
注意,$10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数。

另外,还有几个特殊变量用来处理参数,前面已经提到:

特殊变量说明$#传递给函数的参数个数。$*显示所有传递给函数的参数。$@与$*相同,但是略有区别,请查看Shell特殊变量。$?函数的返回值。


三、Shell输入输出重定向:Shell Here Document,/dev/null文件

Unix 命令默认从标准输入设备(stdin)获取输入,将结果输出到标准输出设备(stdout)显示。一般情况下,标准输入设备就是键盘,标准输出设备就是终端,即显示器。

输出重定向

命令的输出不仅可以是显示器,还可以很容易的转移向到文件,这被称为输出重定向。

命令输出重定向的语法为:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;">$ <span class="sh_keyword" style="color: rgb(255, 48, 48); font-weight: bold;">command</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);">></span> file</li></ol>
这样,输出到显示器的内容就可以被重定向到文件。

例如,下面的命令在显示器上不会看到任何输出:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;">$ who <span class="sh_symbol" style="color: rgb(48, 48, 238);">></span> users</li></ol>
打开 users 文件,可以看到下面的内容:
$ cat usersoko         tty01   Sep 12 07:30ai          tty15   Sep 12 13:32ruth        tty21   Sep 12 10:10pat         tty24   Sep 12 13:07steve       tty25   Sep 12 13:03$
输出重定向会覆盖文件内容,请看下面的例子:
$ echo line 1 > users$ cat usersline 1$
如果不希望文件内容被覆盖,可以使用 >> 追加到文件末尾,例如:
$ echo line 2 >> users$ cat usersline 1line 2$

输入重定向

和输出重定向一样,Unix 命令也可以从文件获取输入,语法为:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_keyword" style="color: rgb(255, 48, 48); font-weight: bold;">command</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);"><</span> file</li></ol>
这样,本来需要从键盘获取输入的命令会转移到文件读取内容。

注意:输出重定向是大于号(>),输入重定向是小于号(<)。

例如,计算 users 文件中的行数,可以使用下面的命令:
$ wc -l users2 users$
也可以将输入重定向到 users 文件:
$ wc -l < users2$
注意:上面两个例子的结果不同:第一个例子,会输出文件名;第二个不会,因为它仅仅知道从标准输入读取内容。

重定向深入讲解

一般情况下,每个 Unix/Linux 命令运行时都会打开三个文件:
  • 标准输入文件(stdin):stdin的文件描述符为0,Unix程序默认从stdin读取数据。
  • 标准输出文件(stdout):stdout 的文件描述符为1,Unix程序默认向stdout输出数据。
  • 标准错误文件(stderr):stderr的文件描述符为2,Unix程序会向stderr流中写入错误信息。

默认情况下,command > file 将 stdout 重定向到 file,command < file 将stdin 重定向到 file。

如果希望 stderr 重定向到 file,可以这样写:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_variable" style="color: rgb(0, 0, 255);">$command</span> <span class="sh_number" style="color: rgb(50, 186, 6);">2</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);">></span> file</li></ol>
如果希望 stderr 追加到 file 文件末尾,可以这样写:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_variable" style="color: rgb(0, 0, 255);">$command</span> <span class="sh_number" style="color: rgb(50, 186, 6);">2</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);">>></span> file</li></ol>
2 表示标准错误文件(stderr)。

如果希望将 stdout 和 stderr 合并后重定向到 file,可以这样写:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_variable" style="color: rgb(0, 0, 255);">$command</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);">></span> file <span class="sh_number" style="color: rgb(50, 186, 6);">2</span><span class="sh_symbol" style="color: rgb(48, 48, 238);">>&</span><span class="sh_number" style="color: rgb(50, 186, 6);">1</span></li></ol>
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_variable" style="color: rgb(0, 0, 255);">$command</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);">>></span> file <span class="sh_number" style="color: rgb(50, 186, 6);">2</span><span class="sh_symbol" style="color: rgb(48, 48, 238);">>&</span><span class="sh_number" style="color: rgb(50, 186, 6);">1</span></li></ol>
如果希望对 stdin 和 stdout 都重定向,可以这样写:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_variable" style="color: rgb(0, 0, 255);">$command</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);"><</span> file1 <span class="sh_symbol" style="color: rgb(48, 48, 238);">></span>file2</li></ol>
command 命令将 stdin 重定向到 file1,将 stdout 重定向到 file2。 

全部可用的重定向命令列表命令说明command > file将输出重定向到 file。command < file将输入重定向到 file。command >> file将输出以追加的方式重定向到 file。n > file将文件描述符为 n 的文件重定向到 file。n >> file将文件描述符为 n 的文件以追加的方式重定向到 file。n >& m将输出文件 m 和 n 合并。n <& m将输入文件 m 和 n 合并。<< tag将开始标记 tag 和结束标记 tag 之间的内容作为输入。

Here Document

Here Document 目前没有统一的翻译,这里暂译为”嵌入文档“。Here Document 是 Shell 中的一种特殊的重定向方式,它的基本的形式如下:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_keyword" style="color: rgb(255, 48, 48); font-weight: bold;">command</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);"><<</span> delimiter</li><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;">    document</li><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;">delimiter</li></ol>
它的作用是将两个 delimiter 之间的内容(document) 作为输入传递给 command。

注意:
  • 结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。
  • 开始的delimiter前后的空格会被忽略掉。

下面的例子,通过 wc -l 命令计算 document 的行数:
$wc -l << EOF    This is a simple lookup program    for good (and bad) restaurants    in Cape Town.EOF3$
也可以 将 Here Document 用在脚本中,例如:

[cpp] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. #!/bin/bash  
  2. cat << EOF  
  3. This is a simple lookup program  
  4. for good (and bad) restaurants  
  5. in Cape Town.  
  6. EOF  
运行结果:

This is a simple lookup programfor good (and bad) restaurantsin Cape Town.

下面的脚本通过 vi 编辑器将 document 保存到 test.txt 文件:
[cpp] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. #!/bin/sh  
  2. filename=test.txt  
  3. vi $filename <<EndOfCommands  
  4. i  
  5. This file was created automatically from  
  6. a shell script  
  7. ^[  
  8. ZZ  
  9. EndOfCommands  
运行脚本:
$ sh test.shVim: Warning: Input is not from a terminal$
打开 test.txt,可以看到下面的内容:
$ cat test.txtThis file was created automatically froma shell script$

/dev/null 文件

如果希望执行某个命令,但又不希望在屏幕上显示输出结果,那么可以将输出重定向到 /dev/null:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;">$ <span class="sh_keyword" style="color: rgb(255, 48, 48); font-weight: bold;">command</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);">></span> <span class="sh_normal">/dev/null</span></li></ol>
/dev/null 是一个特殊的文件,写入到它的内容都会被丢弃;如果尝试从该文件读取内容,那么什么也读不到。但是 /dev/null 文件非常有用,将命令的输出重定向到它,会起到”禁止输出“的效果。

如果希望屏蔽 stdout 和 stderr,可以这样写:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;">$ <span class="sh_keyword" style="color: rgb(255, 48, 48); font-weight: bold;">command</span> <span class="sh_symbol" style="color: rgb(48, 48, 238);">></span> <span class="sh_normal">/dev/null</span> <span class="sh_number" style="color: rgb(50, 186, 6);">2</span><span class="sh_symbol" style="color: rgb(48, 48, 238);">>&</span><span class="sh_number" style="color: rgb(50, 186, 6);">1</span></li></ol>

四、Shell文件包含

像其他语言一样,Shell 也可以包含外部脚本,将外部脚本的内容合并到当前脚本。

Shell 中包含脚本可以使用:

<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_symbol" style="color: rgb(48, 48, 238);">.</span> filename</li></ol>
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_keyword" style="color: rgb(255, 48, 48); font-weight: bold;">source</span> filename</li></ol>
两种方式的效果相同,简单起见,一般使用点号(.),但是注意点号(.)和文件名中间有一空格。

例如,创建两个脚本,一个是被调用脚本 subscript.sh,内容如下:
<ol class="snippet-num" style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; padding-left: 20px; padding-top: 3px !important; padding-right: 8px !important; padding-bottom: 3px !important;"><li style="padding-left: 8px; list-style: decimal-leading-zero outside none;"><span class="sh_variable" style="color: rgb(0, 0, 255);">url</span><span class="sh_symbol" style="color: rgb(48, 48, 238);">=</span><span class="sh_string" style="color: rgb(24, 97, 167);">"http://see.xidian.edu.cn/cpp/view/2738.html"</span></li></ol>
一个是主文件 main.sh,内容如下:

[cpp] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. #!/bin/bash  
  2. . ./subscript.sh  
  3. echo $url  

执行脚本:
$chomd +x main.sh./main.shhttp://see.xidian.edu.cn/cpp/view/2738.html$
注意:被包含脚本不需要有执行权限。
0 0
原创粉丝点击