高级 bash 脚本编程(shell脚本)学习总结

高级 bash 脚本编程指南(shell脚本)

来源于实验楼

1.1       bash介绍与入门

1.1.1        运行Bash脚本的方式

1)  # 使用shell来执行

$ sh hello.sh

2)  # 使用bash来执行

$ bash hello.sh

3)  使用.来执行

$ ./hello.sh

4)  使用source来执行

$ source hello.sh

5)  还可以赋予脚本所有者执行权限,允许该用户执行该脚本

$ chmod u+rx hello.sh

$  ./hello.sh

1.1.2        使用重定向

比如我们想要保存刚刚的hello world为一个文本,那么该怎么办呢? > 这个符号是重定向,执行以下代码,就会在当前目录下生成一个my.txt。打开看看有没有hello world。

#!/bin/bash

 echo "Hello World" > my.txt

1.1.3        使用脚本清除/var/log下的log文件

首先我们看一看/var/log/wtmp里面有啥东西(cat /var/log/wtmp),这个文件中记录了系统的一些信息,现在我们需要写一个脚本把里面的东西清空,但是保留文件。$ vim cleanlogs.sh

说明:/dev/null这个东西可以理解为一个黑洞,里面是空的(可以用cat命令看一看),什么东西都可以往里面扔,扔了就没了。

u  代码

#!/bin/bash

# 初始化一个变量

LOG_DIR=/var/log

cd $LOG_DIR

cat /dev/null > wtmp

echo "Logs cleaned up."

exit

运行脚本前,先看看 /var/log/wtmp 文件内是否有内容。运行此脚本后,文件的内容将被清除。

u  执行

1)  由于脚本中含有对系统日志文件内容的清除操作,这要求要有管理员权限.不然会报permission denide错误,使用sudo命令调用管理员权限才能执行成功:

$ sudo ./cleanlogs.sh

2)  #!/bin/bash这一行是表示使用/bin/bash作为脚本的解释器,这行要放在脚本的行首并且不要省略

3)  脚本正文中以#号开头的行都是注释语句,这些行在脚本的实际执行过程中不会被执行。这些注释语句能方便我们在脚本中做一些注释或标记,让脚本更具可读性。

1.1.4        思考练习

1)  遇到权限不够的提示,为什么,如何解决?

权限不够加sudo啊,可是你会发现sudo cat /dev/null > /var/log/wtmp 一样会提示权限不够,为什么呢?因为sudo只能让cat命令以sudo的权限执行,而对于>这个符号并没有sudo的权限,我们可以使用sudo sh -c "cat /dev/null > /var/log/wtmp " 让整个命令都具有sudo的权限执行。

2)  为什么cleanlogs.sh可以将log文件清除?

因为/dev/null ,里面是空的,什么东西都可以往里面扔,扔了就没了。

1.2       bash特殊字符上

1.2.1        注释(#)

全部特殊字符的作用已经列表如下:

 

行首以 # 开头(除#!之外)的是注释。#!是用于指定当前脚本的解释器,我们这里为bash,且应该指明完整路径,所以为/bin/bash。当然,在echo中转义的 # 是不能作为注释的:

$ vim test.sh

输入如下代码,并保存。(中文为注释,不需要输入)

#!/bin/bash

echo "The # here does not begin a comment."

echo 'The # here does not begin a comment.'

echo The \# here does not begin a comment.

echo The # 这里开始一个注释

echo $(( 2#101011 ))     # 数制转换(使用二进制表示),不是一个注释,双括号表示对于数字的处理

解释说明

上面的脚本说明了如何使用echo打印出一段字符串和变量内容,这里采用了几种不同的方式,希望你可以理解这几种不同方式的异同。

1.2.2        分号(;)

1)  命令分隔符

使用分号(;)可以在同一行上写两个或两个以上的命令。

#!/bin/bash

echo hello; echo there

filename=ttt.sh

if [ -e "$filename" ]; then    # 注意: "if"和"then"需要分隔,-e用于判断文件是否存在

echo "File $filename exists."; cp $filename $filename.bak

else

echo "File $filename not found."; touch $filename

fi; echo "File test complete."

解释说明

上面脚本使用了一个if分支判断一个文件是否存在,如果文件存在打印相关信息并将该文件备份;如果不存在打印相关信息并创建一个新的文件。最后将输出"测试完成"。

1)  终止case选项(双分号)

使用双分号(;;)可以终止case选项。

#!/bin/bash

varname=b

case "$varname" in

[a-z]) echo "abc";;

[0-9]) echo "123";;

esac

解释说明

上面脚本使用case语句,首先创建了一个变量初始化为b,然后使用case语句判断该变量的范围,并打印相关信息。如果你有其它编程语言的经验,这将很容易理解。

1.2.3        点号(.)

等价于 source 命令,bash 中的 source 命令用于在当前 bash 环境下读取并执行 FileName.sh 中的命令。

$ source test.sh

Hello World

$ . test.sh

Hello World

1.2.4        引号

1)  双引号(")

"STRING" 将会阻止(解释)STRING中大部分特殊的字符。

2)  单引号(')

'STRING' 将会阻止STRING中所有特殊字符的解释,这是一种比使用"更强烈的形式。后面的实验会详细说明。

3)  区别

这里举一个例子,能够更加生动的说明:

 

同样是$HOME,单引号会直接认为是字符,而双引号认为是一个变量。

1.2.5        斜线和反斜线

1)  斜线(/)

文件名路径分隔符。分隔文件名不同的部分(如/home/bozo/projects/Makefile)。也可以用来作为除法算术操作符。注意在linux中表示路径的时候,许多个/跟一个/是一样的。

/home/shiyanlou等同于////home///shiyanlou

2)  反斜线(\)

一种对单字符的引用机制。\X 将会“转义”字符X。这等价于"X",也等价于'X'。\ 通常用来转义双引号(")和单引号('),这样双引号和单引号就不会被解释成特殊含义了。

符号说明

u  \n 表示新的一行

u  \r 表示回车

u  \t 表示水平制表符

u  \v 表示垂直制表符

u  \b 表示后退符

u  \a 表示"alert"(蜂鸣或者闪烁)

u  \0xx 转换为八进制的ASCII码, 等价于0xx

u  " 表示引号字面的意思

转义符也提供续行功能,也就是编写多行命令的功能。每一个单独行都包含一个不同的命令,但是每行结尾的转义符都会转义换行符,这样下一行会与上一行一起形成一个命令序列。

1.2.6        反引号(`)

命令替换,反引号中的命令会优先执行,如:

$ cp `mkdir back` test.sh back

$ ls

(先创建了 back 目录,然后复制 test.sh 到 back 目录)

1.2.7        冒号(:)

1)  空命令

等价于“NOP”(no op,一个什么也不干的命令)。也可以被认为与shell的内建命令true作用相同。“:”命令是一个bash的内建命令,它的退出码(exit status)是(0)。如:

#!/bin/bash

while :

do

    echo "endless loop"

done

等价于

#!/bin/bash

while true

do

    echo "endless loop"

done

 

2)  可以在 if/then 中作占位符:

#!/bin/bash

condition=5

if [ $condition -gt 0 ] #gt表示greater than,也就是大于,同样有-lt(小于),-eq(等于)

then :   # 什么都不做,退出分支

else

    echo "$condition"

fi

3)  变量扩展/子串替换

在与>重定向操作符结合使用时,将会把一个文件清空,但是并不会修改这个文件的权限。如果之前这个文件并不存在,那么就创建这个文件。

$ : > test.sh   # 文件“test.sh”现在被清空了

# 与 cat /dev/null > test.sh 的作用相同

# 然而,这并不会产生一个新的进程, 因为“:”是一个内建命令

在与>>重定向操作符结合使用时,将不会对预先存在的目标文件 (: >> target_file)产生任何影响。如果这个文件之前并不存在,那么就创建它。

也可能用来作为注释行,但不推荐这么做。使用 # 来注释的话,将关闭剩余行的错误检查,所以可以在注释行中写任何东西。然而,使用 : 的话将不会这样。如:

$ : This is a comment that generates an error,( if [ $x -eq 3] )

":"还用来在 /etc/passwd 和 $PATH 变量中做分隔符,如:

$ echo $PATH

/usr/local/bin:/bin:/usr/bin:/usr/X11R6/bin:/sbin:/usr/sbin:/usr/games

1.2.8        问号(?)

测试操作符,在一个双括号结构中,? 就是C语言的三元操作符,如:

输入如下代码,并保存:

 #!/bin/bash

 a=10

 (( t=a<50?8:9 ))

 echo $t

1.2.9        美元符号($)

变量替换

$ vim test.sh

#!/bin/bash

var1=5

var2=23skidoo

echo $var1     # 5

echo $var2     # 23skidoo

1.3       bash特殊字符下

1.3.1        小括号(( ))

下面是全部特殊字符的作用的列表:

 

1)  命令组

在括号中的命令列表,将会作为一个子 shell 来运行。在括号中的变量,由于是在子shell中,所以对于脚本剩下的部分是不可用的。父进程,也就是脚本本身,将不能够读取在子进程中创建的变量,也就是在子shell 中创建的变量。如:

#!/bin/bash

a=123

( a=321; )

echo "$a" #a的值为123而不是321,因为括号将判断为局部变量

2)  初始化数组

#!/bin/bash

arr=(1 4 5 7 9 21)

echo ${arr[3]} # get a value of arr 7

 

1.3.2        大括号({ })

1)  文件名扩展

复制 t.txt 的内容到 t.back 中

#!/bin/bash

if [ ! -w 't.txt' ];

then

    touch t.txt

fi

echo 'test text' >> t.txt

cp t.{txt,back}

注意: 在大括号中,不允许有空白,除非这个空白被引用或转义。

2)  代码块

代码块,又被称为内部组,这个结构事实上创建了一个匿名函数(一个没有名字的函数)。然而,与“标准”函数不同的是,在其中声明的变量,对于脚本其他部分的代码来说还是可见的。

#!/bin/bash

a=123

{ a=321; }

echo "a = $a"

运行代码:

$ bash test23.sh

a = 321

变量 a 的值被更改了。

1.3.3        括号([ ])

1)  条件测试

条件测试表达式放在[ ]中。下列练习中的-lt (less than)表示小于号。

#!/bin/bash

a=5

if [ $a -lt 10 ]

then

    echo "a: $a"

else

    echo 'a>10'

fi

双中括号([[ ]])也用作条件测试(判断),后面的实验会详细讲解。

2)  数组元素

在一个array结构的上下文中,中括号用来引用数组中每个元素的编号。

#!/bin/bash

arr=(12 22 32)

arr[0]=10

echo ${arr[0]}

1.3.4        尖括号(< 和 >)

1)  重定向

u  test.sh > filename:重定向test.sh的输出到文件 filename 中。如果 filename 存在的话,那么将会被覆盖。

u  test.sh &> filename:重定向 test.sh 的 stdout(标准输出)和 stderr(标准错误)到 filename 中。

u  test.sh >&2:重定向 test.sh 的 stdout 到 stderr 中。

u  test.sh >> filename:把 test.sh 的输出追加到文件 filename 中。如果filename 不存在的话,将会被创建。

1.3.5  竖线(|

管道:分析前边命令的输出,并将输出作为后边命令的输入。这是一种产生命令链的好方法。

#!/bin/bash

tr 'a-z' 'A-Z'

exit 0

现在让我们输送ls -l的输出到一个脚本中:

$ chmod 755 test26.sh

$ ls -l | ./test26.sh

1.3.6        破折号(-)

1)  选项,前缀

在所有的命令内如果想使用选项参数的话,前边都要加上“-”。

#!/bin/bash

a=5

b=5

if [ "$a" -eq "$b" ]

then

    echo "a is equal to b."

fi

2)  用于重定向stdin或stdout

下面脚本用于备份最后24小时当前目录下所有修改的文件.

#!/bin/bash

BACKUPFILE=backup-$(date +%m-%d-%Y)

# 在备份文件中嵌入时间.

archive=${1:-$BACKUPFILE}

#  如果在命令行中没有指定备份文件的文件名,

#  那么将默认使用"backup-MM-DD-YYYY.tar.gz".

tar cvf - `find . -mtime -1 -type f -print` > $archive.tar

gzip $archive.tar

echo "Directory $PWD backed up in archive file \"$archive.tar.gz\"."

exit 0

1.4       变量和参数

1.4.1        变量定义

1.4.1.1       概念

变量的名字就是变量保存值的地方。引用变量的值就叫做变量替换。如果 variable 是一个变量的名字,那么 $variable 就是引用这个变量的值,即这变量所包含的数据。$variable 事实上只是 ${variable} 的简写形式。在某些上下文中 $variable 可能会引起错误,这时候你就需要用 ${variable} 了。

1.4.1.2        定义变量

定义变量时,变量名不加美元符号($,PHP语言中变量需要),如: myname="shiyanlou"。而且变量名和等号之间不能有空格。同时,变量名的命名须遵循如下规则:

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

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

u  不能使用标点符号。

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

除了直接赋值,还可以用语句给变量赋值,如:for file in `ls /etc`

1.4.2        使用变量

变量名前加美元符号,如:

myname="shiyanlou"
echo $myname
echo ${myname}
echo ${myname}Good
echo $mynameGood
myname="miao"
echo ${myname}

花括号帮助解释器识别变量的边界,若不加,解释器会把mynameGood当成一个变量(值为空)

u  推荐给所有变量加花括号

u  已定义的变量可以重新被定义

1.4.3        只读变量

使用 readonly 命令可以将变量定义为只读变量,只读变量的值不能被改变。 下面的例子尝试更改只读变量,结果报错:

#!/bin/bash
myUrl="http://www.shiyanlou.com"
readonly myUrl
myUrl="http://www.shiyanlou.com"

运行脚本,结果如下:

/bin/sh: NAME: This variable is read only.

1.4.4        特殊变量

1.4.4.1        局部变量

这种变量只有在代码块或者函数中才可见。

1.4.4.2        环境变量

这种变量将影响用户接口和 shell 的行为。在通常情况下,每个进程都有自己的“环境”,这个环境是由一组变量组成的,这些变量中存有进程可能需要引用的信息。在这种情况下,shell 与一个一般的进程没什么区别。

1.4.4.3        位置参数

从命令行传递到脚本的参数:$0,$1,$2,$3...

$0就是脚本文件自身的名字,$1 是第一个参数,$2 是第二个参数,$3 是第三个参数,然后是第四个。$9 之后的位置参数就必须用大括号括起来了,比如,${10},${11},${12}。

u  $# : 传递到脚本的参数个数

u  $* : 以一个单字符串显示所有向脚本传递的参数。与位置变量不同,此选项参数可超过 9个

u  $$ : 脚本运行的当前进程 ID号

u  $! : 后台运行的最后一个进程的进程 ID号

u  $@ : 与$#相同,但是使用时加引号,并在引号中返回每个参数

u  $: 显示shell使用的当前选项,与 set命令功能相同

u  $? : 显示最后命令的退出状态。 0表示没有错误,其他任何值表明有错误。

1.4.4.4        位置参数实例

这个十分重要,在我们运行一套脚本的时候,有时候是需要参数的,这里我们教大家如何获取参数

#!/bin/bash
# 作为用例, 调用这个脚本至少需要10个参数, 比如:
# bash test.sh 1 2 3 4 5 6 7 8 9 10
MINPARAMS=10
echo
echo "The name of this script is \"$0\"."
echo "The name of this script is \"`basename $0`\"."
echo
if [ -n "$1" ]              # 测试变量被引用.
then
echo "Parameter #1 is $1"  # 需要引用才能够转义"#"
fi 
if [ -n "$2" ]
then
echo "Parameter #2 is $2"
fi 
if [ -n "${10}"# 大于$9的参数必须用{}括起来.
then
echo "Parameter #10 is ${10}"
fi 
echo "-----------------------------------"
echo "All the command-line parameters are: "$*""
if [ $# -lt "$MINPARAMS" ]
then
 echo
 echo "This script needs at least $MINPARAMS command-line arguments!"
fi  
echo
exit 0

运行代码:

$ bash test30.sh 1 2 10
The name of this script is "test.sh".
The name of this script is "test.sh".
Parameter #1 is 1
Parameter #2 is 2
-----------------------------------
All the command-line parameters are: 1 2 10
This script needs at least 10 command-line arguments!
 

1.5       基本运算符

1.5.1        算数运算符

 

#!/bin/bash

a=10

b=20

val=`expr $a + $b`

echo "a + b : $val"

val=`expr $a - $b`

echo "a - b : $val"

val=`expr $a \* $b`

echo "a * b : $val"

val=`expr $b / $a`

echo "b / a : $val"

val=`expr $b % $a`

echo "b % a : $val"

if [ $a == $b ]

then

   echo "a == b"

fi

if [ $a != $b ]

then

   echo "a != b"

fi

原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用。expr 是一款表达式计算工具,使用它能完成表达式的求值操作。注意使用的反引号(esc键下边)。

u  表达式和运算符之间要有空格$a + $b写成$a+$b不行

u  条件表达式要放在方括号之间,并且要有空格[ $a == $b ]写成[$a==$b]不行

u  乘号(*)前边必须加反斜杠(\)才能实现乘法运算

1.5.2        关系运算符

关系运算符只支持数字,不支持字符串,除非字符串的值是数字。

 

#!/bin/bash

a=10

b=20

if [ $a -eq $b ]

then

   echo "$a -eq $b : a == b"

else

   echo "$a -eq $b: a != b"

fi

1.5.3        逻辑运算符

 

#!/bin/bash

a=10

b=20

if [[ $a -lt 100 && $b -gt 100 ]]

then

   echo "return true"

else

   echo "return false"

fi

if [[ $a -lt 100 || $b -gt 100 ]]

then

   echo "return true"

else

   echo "return false"

fi

1.5.4        字符串运算符

 

#!/bin/bash

a="abc"

b="efg"

if [ $a = $b ]

then

   echo "$a = $b : a == b"

else

   echo "$a = $b: a != b"

fi

if [ -n $a ]

then

   echo "-n $a : The string length is not 0"

else

   echo "-n $a : The string length is  0"

fi

if [ $a ]

then

   echo "$a : The string is not empty"

else

   echo "$a : The string is empty"

fi

1.5.5        文件测试运算符

 

#!/bin/bash

file="/home/shiyanlou/test.sh"

if [ -r $file ]

then

   echo "The file is readable"

else

   echo "The file is not readable"

fi

if [ -e $file ]

then

   echo "File exists"

else

   echo "File not exists"

fi

1.6       流程控制

1.6.1        if else

和Java、PHP等语言不一样,sh的流程控制不可为空在sh/bash里可不能这么写,如果else分支没有语句执行,就不要写这个else。

1)   if

if 语句语法格式:

if condition

then

    command1

    command2

    ...

    commandN

fi

2)   if else

if else 语法格式:

if condition

then

    command1

    command2

    ...

    commandN

else

    command

fi

3)   if-elif-else 语法格式:

if condition1

then

    command1

elif condition2

then

    command2

else

    commandN

fi

u  以下实例判断两个变量是否相等:

a=10

b=20

if [ $a == $b ]

then

   echo "a == b"

elif [ $a -gt $b ]

then

   echo "a > b"

elif [ $a -lt $b ]

then

   echo "a < b"

else

   echo "Ineligible"

fi

u  if else语句经常与test命令结合使用

num1=$[2*3]

num2=$[1+5]

if test $[num1] -eq $[num2]

then

    echo 'Two numbers are equal!'

else

    echo 'The two numbers are not equal!'

fi

1.6.2        for 循环

u  for循环一般格式为:

for var in item1 item2 ... itemN

do

    command1

    command2

    ...

    commandN

done

u  例如,顺序输出当前列表中的数字:

for loop in 1 2 3 4 5

do

    echo "The value is: $loop"

done

输出结果:

The value is: 1

The value is: 2

The value is: 3

The value is: 4

The value is: 5

顺序输出字符串中的字符:

for str in This is a string

do

    echo $str

done

输出结果:

This

is

a

string

1.6.3        while 语句

while循环用于不断执行一系列命令,也用于从输入文件中读取数据;命令通常为测试条件。其格式为:

while condition

do

    command

done

实例:

#!/bin/bash

int=1

while(( $int<=5 ))

do

    echo $int

    let "int++"

done

如果int小于等于5,那么条件返回真。int从1开始,每次循环处理时,int加1。运行上述脚本,返回数字1到5,然后终止。使用了 Bash let 命令,它用于执行一个或多个表达式,变量计算中不需要加上 $ 来表示变量。

while循环可用于读取键盘信息。下面的例子中,输入信息被设置为变量MAN,按结束循环。

echo 'press <CTRL-D> exit'

echo -n 'Who do you think is the most handsome: '

while read MAN

do

    echo "Yes!$MAN is really handsome"

done

1.6.4        无限循环

无限循环语法格式:

while :

do

    command

done

或者

while true

do

    command

done

或者

for (( ; ; ))

1.6.5        until 循环

until循环执行一系列命令直至条件为真时停止。 until循环与while循环在处理方式上刚好相反。 一般while循环优于until循环,但在某些时候—也只是极少数情况下,until循环更加有用。 until 语法格式:

until condition

do

    command

done

条件可为任意测试条件,测试发生在循环末尾,因此循环至少执行一次—请注意这一点。

1.6.6        case

Shell case语句为多选择语句。可以用case语句匹配一个值与一个模式,如果匹配成功,执行相匹配的命令。case语句格式如下:

case in

模式1)

    command1

    command2

    ...

    commandN

    ;;

模式2

    command1

    command2

    ...

    commandN

    ;;

esac

取值后面必须为单词in,每一模式必须以右括号结束。取值可以为变量或常数。匹配发现取值符合某一模式后,其间所有命令开始执行直至 ;;。取值将检测匹配的每一个模式。一旦模式匹配,则执行完匹配模式相应命令后不再继续其他模式。如果无一匹配模式,使用星号 * 捕获该值,再执行后面的命令。

下面的脚本提示输入1到4,与每一种模式进行匹配:

echo 'Enter a number between 1 and 4:'

echo 'The number you entered is:'

read aNum

case $aNum in

    1)  echo 'You have chosen 1'

    ;;

    2)  echo 'You have chosen 2'

    ;;

    3)  echo 'You have chosen 3'

    ;;

    4)  echo 'You have chosen 4'

    ;;

    *)  echo 'You did not enter a number between 1 and 4'

    ;;

esac

1.6.7        跳出循环

在循环过程中,有时候需要在未达到循环结束条件时强制跳出循环,Shell使用两个命令来实现该功能:break和continue。

1)   break命令

break命令允许跳出所有循环(终止执行后面的所有循环)。 下面的例子中,脚本进入死循环直至用户输入数字大于5。要跳出这个循环,返回到shell提示符下,需要使用break命令。

#!/bin/bash

while :

do

    echo -n "Enter a number between 1 and 5:"

    read aNum

    case $aNum in

        1|2|3|4|5) echo "The number you entered is $aNum!"

        ;;

        *) echo "The number you entered is not between 1 and 5! game over!"

            break

        ;;

    esac

done

2)   continue

continue命令与break命令类似,只有一点差别,它不会跳出所有循环,仅仅跳出当前循环。 对上面的例子进行修改:

#!/bin/bash

while :

do

    echo -n "Enter a number between 1 and 5: "

    read aNum

    case $aNum in

        1|2|3|4|5) echo "The number you entered is $aNum!"

        ;;

        *) echo "The number you entered is not between 1 and 5!"

            continue

            echo "game over"

        ;;

    esac

done

运行代码发现,当输入大于5的数字时,该例中的循环不会结束,语句 echo "Game is over!" 永远不会被执行。

1.6.8        esac

case的语法和C family语言差别很大,它需要一个esac(就是case反过来)作为结束标记,每个case分支用右圆括号,用两个分号表示break

1.7       函数

1.7.1        函数定义

shell中函数的定义格式如下:

[ function ] funname [()]

{

    action;

    [return int;]

}

说明:可以带function fun() 定义,也可以直接fun() 定义,不带任何参数。参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。 return后跟数值n(0-255)。

u  下面的例子定义了一个函数并进行调用:

#!/bin/bash

demoFun(){

    echo "This is my first shell function!"

}

echo "-----Execution-----"

demoFun

echo "-----Finished-----"

u  下面定义一个带有return语句的函数:

#!/bin/bash

funWithReturn(){

    echo "This function will add the two numbers of the input..."

    echo "Enter the first number: "

    read aNum

    echo "Enter the second number: "

    read anotherNum

    echo "The two numbers are $aNum and $anotherNum !"

    return $(($aNum+$anotherNum))

}

funWithReturn

echo "The sum of the two numbers entered is $? !"

u  函数返回值在调用该函数后通过 $? 来获得

u  所有函数在使用前必须定义。

1.7.2        函数参数

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

#!/bin/bash

funWithParam(){

    echo "The first parameter is $1 !"

    echo "The second parameter is $2 !"

    echo "The tenth parameter is $10 !"

    echo "The tenth parameter is ${10} !"

    echo "The eleventh parameter is ${11} !"

    echo "The total number of parameters is $# !"

    echo "Outputs all parameters as a string $* !"

}

funWithParam 1 2 3 4 5 6 7 8 9 34 73

注意: $10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数。

猜你喜欢

转载自www.cnblogs.com/myxcf/p/9476242.html
今日推荐