Linnux 中有些命令的功能非常强大,主要是因为它支持的命令选项比较多。如:【ip】命令可以配置IP地址、路由条目的配置管理操作非常完善,该命令就可以完成【ifconfig】和【route】命令实现的所有功能。函数是单独的功能模块,如果函数能够接收选项参数,那么该函数的功能就变得丰富,且灵活。脚本也是一样。
那么linux中命令的格式又是怎么样的呢?
1
2
|
[root@Node1 ~]
# command [optons parameter1 | parameter2]... parameter3 parameter3 ...
命令 选项 选项参数 命令参数(1) 命令参数(2)
|
我们书写脚本的标准:与linux中命令的格式几乎一样。那么如果我们的脚本支持选项参数,格式如下:
1
2
|
scriptName {-option|--option parameter}... scriptParameter ...
脚本名称 短选项 长选项 选项参数 脚本参数
|
说明:
1
2
3
|
option 使用单杠(-)或双杠(--)来表示选项。
parameter 选项参数
scriptParameter 脚本参数
|
这些选项、选项参数和脚本参数都统称为位置参数(positional parameter),shell会使用特殊的变量来记录它们。所以在脚本或函数中就可以使用这些变量来引用脚本或函数的参数了。既然称为位置参数,应该有前后顺序。那么bash shell是如何标识不同位置的参数的呢?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
$1 该变量记录第1个参数
$2 该变量记录第2个参数
$3 该变量记录第3个参数
... ....
$9 该变量记录第9个参数
${10} 该变量记录第10个参数
... ...
$0 在命令行运行脚本时:
如果使用绝对路径运行脚本的话,记录脚本名称时使用的是绝对路径。
如果使用相对路径运行脚本的话,记录脚本名称时使用的是相对路径。
可以使用: 【
basename
$0】方法去除脚本的路径。只获取脚本名称。
$
# 位置参数的个数
$* 获取所有参数,把这些参数当做一个单词处理。不允许使用
for
进行迭代。
$@ 也是获取所有参数。把它当作一个字符串。该字符串中包含多个使用空间分
隔的单词组成,这种方式允许使用
for
迭代出该字符串中的
每一个单词(位置参数)的。
|
那么,我们在脚本或函数中怎么样获取选项参数呢?
分析:书写脚本或函数时,如果支持选项参数的话,要遵循上述格式。选项option有一个特点:都使用单杠(-)或双杠(--)来表示这就是脚本或函数的选项。它后面跟就是选项参数了。这么是如何实现的呢?实现方式有两种:
1、使用 if 判断语句。
1
2
3
|
if
[[
"$1"
=~ ^- ]];
then
$2就是选项$1的参数了
fi
|
2、通过选择结构的方式: case
1
2
3
4
5
|
case
$1
in
-a|--add)
$2就是选项$1的参数了
;;
esac
|
这种获取脚本或函数的选项参数的方式有太多的局限性。对于获取只有一个选项的脚本或函数有用。对于多个选项参数的脚本/函数就无能为力了。
虽然使用if的多支语句可以获取多个选项的参数。要把每个选项都作为一条if的分枝语句,很繁琐。
如果,我们的脚本或函数中要使用到多个选项。那么怎么获取选项的各个参数呢?
bash shell 提供【shift】命令来帮助操作命令行参数,该命令能够改变命令行参数的相对位置。shift 默认将每个参数变量左移一个位置。
如:假如给脚本或函数传递了2个参数,每执行一次shift命令,变量$2的值移组变量$1,变量$1的值被丢弃。
shift 命令的使用
1
|
shift
:
shift
[n] 设置步长。默认是为1 .
|
我们使用【shift】命令与选择语句 case 就可以很方便的取得每一个选项的参数了,而且也很简洁。例:
mkscript.sh [-D|--description "script description"] [-A|--author "script author"] /pa/file
1
2
3
4
5
6
7
8
9
10
11
12
|
case
$1
in
-D|--description)
该就选项的参数就是$2了
shift
2
;;
-A|--author)
该选项的参数就是$2了
shift
2
;;
*)
echo
"Usage: 脚本或函数的使用方法"
esac
|
上述提取,脚本/函数的选项参数的方法有问题:
无论运行脚本mkscript时传递的参数正确也否都会显示"echo "Usage: 脚本或函数的使用方法".都会执行这一步。
最理想的就是:如果用户使用该脚本的时候且带了选项,如果“选项”不是 case 语句列出的“选项”就告诉用户使用该脚本或函数的方法。又能正确无误地取得脚本参数/pa/file.
分析:
执行 case 选择语句,要有一个条件:如果“参数”是"选项"才执行case语句。在脚本中"选项"的标志是:单杠(-)或双杠(--),我们就可以根据该标志判断 case 选择语句是否执行。
因为,我们的脚本选项很多,所以要使用到循环.判断是否是”选项“作为循环的条件。我们把提取脚本的选项参数和脚本参数方法修改成如下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
while
[
"$1"
!=
"${1##[--,-]}"
];
do
case
$1
in
-D|--description)
该选项的参数就是$2
shift
2
;;
-A|--author)
该选项的参数就是$2
shift
2
;;
*)
使用该脚本时,虽然使用了选项,但是脚本不支持的,都匹配 *
在这里就可以提示用户:脚本的使用方法了
shift
2
;;
esac
done
脚本的参数=$1
|