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

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注