在Linux命令中经常带有参数例如[-option]等等。在命令行中可 能有0个或者多个这些选项。我们在之前学习了位置参数,包括$1,$2,$3…,$*,$#,参见Linux Bash Shell学习(七):shell编程基础——运行Shell脚本、function 。这些位置参数都是只读。
移位Shift
shift提供对只读的位置参数的移位赋值的操作,将1=$2,2=$3,…,可以使用shift N来制定移位的数目,例如shift 3,则表示1=$4,2=$5,…。如果命令行中有[-options]的,我们可以对他们进行判断,并进行移位处理。一个简单例子如下:
if [ $1 = -o ]; then [process the -o option] shift
fi [normal processing of arguments...]
下面给两个例子,介绍如何用shift来检查参数。
#例子一:有可能有-N的参数,且为第一参数。如果第一个参数是 -N,记录在howmany中,进行shift,将$2置于$1的位置,如果第一个参数是-X,但X不是数字,给出警告语句,否则howmany使用缺省 的-10。
function test1
{
#对于grep,^表示从匹配行首,$表示匹配行尾,在这里表 示整个$1匹配,而不是部分匹配,*表示前面的字符匹配0个或者多个,下面[0-9]*即表示后面跟着0个或者多个数字,[0-9][0-9]*表示数字 后面有0个或多个数字。故echo $1 | grep '^-[0-9][0-9]*$'表示匹配格式-N,N为数字。grep的具体用法看参见http://www.yesadmin.com/301/135287/index.html
#$(command)表示命令执行的内容。
#-n str,字符串不为null,长度大于零
#请注意双引号的使用,表示这是一个需要检查的str
if [ -n "$(echo $1 | grep '^-[0-9][0-9]*$')" ]; then
howmany=$1
shift elif [ -n "$(echo $1 | grep '^-')" ]; then
echo 'usage: test1 [-N] filename'
exit 1
else
howmany="-10"
fi
echo "sort -nr $1 | head $howmany"
}
test1 -5 $file
对于多个参数,也假设只能放在其他参数之前,如下。对于-b的选项,带自己的参数即[-b barg],对于-b需要多移一位。
while [ -n "$(echo $1 | grep '-')" ]; do
case $1 in
-a ) [process option -a] ;;
-b ) [process option –b, $2 is the option's argument]
shift ;;
-c ) [process option -c] ;;
* ) echo 'usage: alice [-a] [-b barg] [-c] args...'
exit 1
esac
shift
done
[normal processing of arguments...]
但是这种方式不能解决下面两个问题:一、无法使用-abc来代替-a –b -c,二、无法使用-barg来代替-b arg,但是这两种方式在Linux命令中都是比较常见的。我们可以使用getopts这个built-in的命令来处理:
getopts
getopts有两个参数,第一个参数是一个字符串,包括字符和“:”,每一个字符都是一个有效的选项,如果字符后面带有“:”,表示这个字符 有自己的参数。getopts从命令中获取这些参数,并且删去了“-”,并将其赋值在第二个参数中,如果带有自己参数,这个参数赋值在“OPTARG” 中。下面是例子采用getopts的写法。我们将它写成一个可执行的例子。
echo "---Test 3---" function test3
{
echo "test3 $@" #OPTIND记录准备分析(即下一次分析)的第几个参数,如 果我们不使用function,而是直接作为脚本,每次执行OPTIND将重新初始化,但是如果我们使用function的方式,OPTIND将保持上次 的值,即getopts接着分析第N个参数,例如第5个参数,所以需要unset,以保证正确。如果我们用-ab表示-a -b,当分析完-a之后,OPTIND并不加一,保持原值,下次仍然分析该位置参数,如果-b arg,则OPTIND加二。所以我们不需要在getopts中进行shift,可以在完成所有getopts分析后,统一进行shift $(($OPTIND - 1)) echo "OPTIND=$OPTIND" unset OPTIND
#如果我们使用"ab:c",如果option并非所需,则报 告illegal option —d之类的,如果我们去掉相关信息,getopts的第一个参数以":"开头即可。当然也可以通过设置环境参数OPTERR为0,如果不匹配,则将获取的 参数值置为"?",在下面中case的第四个选择可以为\?)或者*)。
while getopts ":ab:c" opt ; do
echo -n "OPTIND=$OPTIND : " case $opt in
a ) echo "process option -$opt" ;;
b ) echo "process option -$opt, $OPTARG is the option's argument" ;;
c ) echo "process option -$opt" ;;
#下面等同于 *) ,因为将opt设置为"?" \? ) echo "process option -$opt"
echo 'usage: alice [-a] [-b barg] [-c] args...'
exit 1
esac
done
echo "OPTIND=$OPTIND, shift $(($OPTIND - 1))"
shift $(($OPTIND - 1)) echo "test $@"
echo "normal processing of arguments..."
}
test3 -b hello -a -c ok
echo
test3 -bhello -ac ok
echo
test3 -ab hello ok
echo
test3 -d
如果命令的参数都采用选项的方式,getopts就可以提供很灵活的分析方式。另外对于非匹配选项,也即"?"的选项,也可以获取命令的参数, 可以根据$((OPTIND-1))获得该位置参数的位置,继而获取该参数的具体内容。