linux shell 移位,Linux Bash Shell学习(十四):命令行选项

在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))获得该位置参数的位置,继而获取该参数的具体内容。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值