20201105记录: Shell $0, $#, $*, $@, $?, $$和命令行参数的使用

特殊变量列表

变量含义
$0 当前脚本的文件名
$n 传递给脚本或函数的参数。n 是一个数字,表示第几个参数。例如,第一个参数是$1,第二个参数是$2。
$# 传递给脚本或函数的参数个数。
$* 传递给脚本或函数的所有参数。
$@ 传递给脚本或函数的所有参数。被双引号(" ")包含时,与 $* 稍有不同,下面将会讲到。
$? 上个命令的退出状态,或函数的返回值。
$$ 当前Shell进程ID。对于 Shell 脚本,就是这些脚本所在的进程ID。

命令行参数

运行脚本时传递给脚本的参数称为命令行参数。命令行参数用 $n 表示,例如,$1 表示第一个参数,$2 表示第二个参数,依次类推。

请看下面的脚本:

1 #!/bin/bash
2 echo "File Name: $0"
3 echo "First Parameter : $1"
4 echo "First Parameter : $2"
5 echo "Quoted Values: $@"
6 echo "Quoted Values: $*"
7 echo "Total Number of Parameters : $#"

运行结果:

1 $./test.sh Zara Ali
2 File Name : ./test.sh
3 First Parameter : Zara
4 Second Parameter : Ali
5 Quoted Values: Zara Ali
6 Quoted Values: Zara Ali
7 Total Number of Parameters : 2

$* 和 $@ 的区别

$* 和 $@ 都表示传递给函数或脚本的所有参数,不被双引号(" ")包含时,都以"$1" "$2" … "$n" 的形式输出所有参数。

但是当它们被双引号(" ")包含时,"$*" 会将所有的参数作为一个整体,以"$1 $2 … $n"的形式输出所有参数;"$@" 会将各个参数分开,以"$1" "$2" … "$n" 的形式输出所有参数。

下面的例子可以清楚的看到 $* 和 $@ 的区别:

 1 #!/bin/bash
 2 echo "\$*=" $*
 3 echo "\"\$*\"=" "$*"
 4 echo "\$@=" $@
 5 echo "\"\$@\"=" "$@"
 6 echo "print each param from \$*"
 7 for var in $*
 8 do
 9 echo "$var"
10 done
11 echo "print each param from \$@"
12 for var in $@
13 do
14 echo "$var"
15 done
16 echo "print each param from \"\$*\""
17 for var in "$*"
18 do
19 echo "$var"
20 done
21 echo "print each param from \"\$@\""
22 for var in "$@"
23 do
24 echo "$var"
25 done

执行 ./test.sh "a" "b" "c" "d",看到下面的结果:

 1 $*=  a b c d
 2 "$*"= a b c d
 3 $@=  a b c d
 4 "$@"= a b c d
 5 print each param from $*
 6 a
 7 b
 8 c
 9 d
10 print each param from $@
11 a
12 b
13 c
14 d
15 print each param from "$*"
16 a b c d
17 print each param from "$@"
18 a
19 b
20 c
21 d

退出状态

$? 可以获取上一个命令的退出状态。所谓退出状态,就是上一个命令执行后的返回结果。

退出状态是一个数字,一般情况下,大部分命令执行成功会返回 0,失败返回 1。

不过,也有一些命令返回其他值,表示不同类型的错误。

下面例子中,命令成功执行:

1 $./test.sh Zara Ali
2 File Name : ./test.sh
3 First Parameter : Zara
4 Second Parameter : Ali
5 Quoted Values: Zara Ali
6 Quoted Values: Zara Ali
7 Total Number of Parameters : 2
8 $echo $?
9 0

介绍下Shell中的${}、##和%%使用范例,本文给出了不同情况下得到的结果。

假设定义了一个变量为:

代码如下:

 1 file=/dir1/dir2/dir3/my.file.txt 

可以用${ }分别替换得到不同的值:
${file#*/}:删掉第一个 / 及其左边的字符串:dir1/dir2/dir3/my.file.txt
${file##*/}:删掉最后一个 /  及其左边的字符串:my.file.txt
${file#*.}:删掉第一个 .  及其左边的字符串:file.txt
${file##*.}:删掉最后一个 .  及其左边的字符串:txt
${file%/*}:删掉最后一个  /  及其右边的字符串:/dir1/dir2/dir3
${file%%/*}:删掉第一个 /  及其右边的字符串:(空值)
${file%.*}:删掉最后一个  .  及其右边的字符串:/dir1/dir2/dir3/my.file
${file%%.*}:删掉第一个  .   及其右边的字符串:/dir1/dir2/dir3/my

记忆的方法为:

# 是 去掉左边(键盘上#在 $ 的左边)
%是去掉右边(键盘上% 在$ 的右边)
单一符号是最小匹配;两个符号是最大匹配
${file:0:5}:提取最左边的 5 个字节:/dir1
${file:5:5}:提取第 5 个字节右边的连续5个字节:/dir2
也可以对变量值里的字符串作替换:
${file/dir/path}:将第一个dir 替换为path:/path1/dir2/dir3/my.file.txt
${file//dir/path}:将全部dir 替换为 path:/path1/path2/path3/my.file.txt

1、执行脚本时是在一个子shell环境运行的,脚本执行完后该子shell自动退出;

2、一个shell中的系统环境变量怎样才会被复制到子shell中(用export定义的变量);

3、一个shell中的系统环境变量只对该shell或者它的子shell有效,该shell结束时变量消失(并不能返回到父shell中)。

4、不用export定义的变量只对该shell有效,对子shell是无效的。

-d :判断制定的是否为目录
-z:判断制定的变量是否存在值
-f:判断制定的是否为文件
-L:判断制定的是否为符号链接
-r:判断制定的是否可读
-s:判断存在的对象长度是否为0
-w:判断制定的是否可写
-x:判断存在的对象是否可以执行
!:测试条件的否定符号
这些文件操作很多时候给脚本编程带来方便,尤其是用在if条件语句中

到此这篇关于Shell $0, $#, $*, $@, $?, $$和命令行参数使用就介绍到这,谢谢。

 

 

 

 

 

 

 

posted @ 2020-11-05 10:57  木笙  阅读(455)  评论(0)    收藏  举报