Shell十三问之九:$@与$*差在哪

要说 $@ 与 $* 之前,需得先从 shell script 的 positional parameter 谈起...

我们都已经知道变量(variable)是如何定义及替换的,这个不用再多讲了。

但是,我们还需要知道有些变量是 shell 内定的,且其名称是我们不能随意修改的,

其中就有 positional parameter 在内。

在 shell script 中,我们可用 $0, $1, $2, $3 ... 这样的变量分别提取命令行中的如下部份:

代码:

 

script_name parameter1 parameter2 parameter3 ...

我们很容易就能猜出 $0 就是代表 shell script 名称(路径)本身,而 $1 就是其后的第一个参数,如此类推....

须得留意的是 IFS 的作用,也就是,若 IFS 被 quoting 处理后,那么 positional parameter 也会改变。

如下例:

代码:

 

my.sh p1 "p2 p3" p4

由于在 p2 与 p3 之间的空格键被 soft quote 所关闭了,因此 my.sh 中的 $2 是 "p2 p3" 而 $3 则是 p4 ...

还记得前两章我们提到 fucntion 时,我不是说过它是 script 中的 script 吗? ^_^

是的,function 一样可以读取自己的(有别于 script 的) postitional parameter ,惟一例外的是 $0 而已。

举例而言:假设 my.sh 里有一个 fucntion 叫 my_fun , 若在 script 中跑 my_fun fp1 fp2 fp3 ,

那么,function 内的 $0 是 my.sh ,而 $1 则是 fp1 而非 p1 了...

不如写个简单的 my.sh script 看看吧:

代码:

 

#!/bin/bash

my_fun() {

    echo '$0 inside function is '$0

    echo '$1 inside function is '$1

    echo '$2 inside function is '$2

}

echo '$0 outside function is '$0

echo '$1 outside function is '$1

echo '$2 outside function is '$2

my_fun fp1 "fp2 fp3"

然后在 command line 中跑一下 script 就知道了:

代码:

 

chmod +x my.sh

./my.sh p1 "p2 p3"

$0 outside function is ./my.sh

$1 outside function is p1

$2 outside function is p2 p3

$0 inside function is ./my.sh

$1 inside function is fp1

$2 inside function is fp2 fp3

然而,在使用 positional parameter 的时候,我们要注意一些陷阱哦:

* $10 不是替换第 10 个参数,而是替换第一个参数($1)然后再补一个 0 于其后﹗

也就是,my.sh one two three four five six seven eigth nine ten 这样的 command line ,

my.sh 里的 $10 不是 ten 而是 one0 哦... 小心小心﹗

要抓到 ten 的话,有两种方法:

方法一是使用我们上一章介绍的 ${ } ,也就是用 ${10} 即可。

方法二,就是 shift 了。

用通俗的说法来说,所谓的 shift 就是取消 positional parameter 中最左边的参数( $0 不受影响)。

其默认值为 1 ,也就是 shift 或 shift 1 都是取消 $1 ,而原本的 $2 则变成 $1、$3 变成 $2 ...

若 shift 3 则是取消前面三个参数,也就是原本的 $4 将变成 $1 ...

那,亲爱的读者,你说要 shift 掉多少个参数,才可用 $1 取得 ${10} 呢? ^_^

okay,当我们对 positional parameter 有了基本概念之后,那再让我们看看其它相关变量吧。

首先是 $# :它可抓出 positional parameter 的数量。

以前面的 my.sh p1 "p2 p3" 为例:

由于 p2 与 p3 之间的 IFS 是在 soft quote 中,因此 $# 可得到 2 的值。

但如果 p2 与 p3 没有置于 quoting 中话,那 $# 就可得到 3 的值了。

同样的道理在 function 中也是一样的...

因此,我们常在 shell script 里用如下方法测试 script 是否有读进参数:

代码:

 

[ $# = 0 ]

假如为 0 ,那就表示 script 没有参数,否则就是有带参数...

接下来就是 $@ 与 $* :

精确来讲,两者只有在 soft quote 中才有差异,否则,都表示"全部参数"( $0 除外)。

举例来说好了:

若在 command line 上跑 my.sh p1 "p2 p3" p4 的话,

不管是 $@ 还是 $* ,都可得到 p1 p2 p3 p4 就是了。

但是,如果置于 soft quote 中的话:

"$@" 则可得到 "p1" "p2 p3" "p4" 这三个不同的词段(word)﹔

"$*" 则可得到 "p1 p2 p3 p4" 这一整串单一的词段。

我们可修改一下前面的 my.sh ,使之内容如下:

代码:

 

#!/bin/bash

my_fun() {

    echo "$#"

}

echo 'the number of parameter in "$@" is '$(my_fun "$@")

echo 'the number of parameter in "$*" is '$(my_fun "$*")

然后再执行 ./my.sh p1 "p2 p3" p4 就知道 $@ 与 $* 差在哪了 ... ^_^

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章