在Linux bash中,可以用以下三种方式解析命令行参数:

  1. 直接处理:使用$1,$2,$3…进行解析

  2. getopts: 单个字符选项的情况,例如:-n 10 -f file.txt等选项

  3. getopt:处理单个字符或长选项(long-option),例如:–prefix=/home等

1. 直接处理

$0   #即命令本身,相当于c/c++中的argv[0]
$1   #第一个参数
$2, $3, $4 ...   #第2、3、4个参数,依次类推
$#   #参数的个数,不包括命令本身
$@   #参数本身的列表,不包括命令本身
$*   #和$@相同,但"$*"和"$@"(加引号)并不同,
     #"$*"将所有的参数解释成一个字符串,而"$@"是一个参数数组

2. getopts

  • getopts时bash的内部命令

  • getopts有两个参数,第一个参数是一个字符串,包括字符和“:”

  • 每一个字符都是一个有效选项(option),如果字符后面带有“:”,表示这个选项有自己的argument,argument保存在内置变量OPTARG中

  • $OPTARG总是存储原始$*中下一个要处理的元素位置

  • 对于while getopts “:abc” opt,第一个冒号表示忽略错误

getopts.sh
#!/bin/bash

echo "原始参数:[$*]"
echo "原始OPTIND:[$OPTIND]"
while getopts ":a:bc" opt
do
    case $opt in
        a)
            echo "-a选项:OPTARG=[$OPTARG] OPTIND=[$OPTIND]"
            ;;
        b)
            echo "-b选项:OPTARG=[$OPTARG] OPTIND=[$OPTIND]"
            ;;
        c)
            echo "-c选项:OPTARG=[$OPTARG] OPTIND=[$OPTIND]"
            ;;
        ?)
            echo "未识别的参数"
            exit 1
            ;;
    esac
done
#通过shift $(($OPTIND - 1))的处理,$*中就只保留了除去选项内容的参数,
#可以在后面的shell程序中进行处理
shift $(($OPTIND - 1))

echo "剩余的参数:[$*]"
echo "\$1=[$1]"
echo "\$2=[$2]"
执行结果
bash getopts.sh -a aaaa -b -c dddd eeee
原始参数:[-a aaaa -b -c dddd eeee]
原始OPTIND:[1]
-a选项:OPTARG=[aaaa] OPTIND=[3]
-b选项:OPTARG=[] OPTIND=[4]
-c选项:OPTARG=[] OPTIND=[5]
剩余的参数:[dddd eeee]
$1=[dddd]
$2=[eeee]

3. getopt

  • getopt是一个外部命令,通常Linux发行版会自带

  • getopt支持短选项和长选项

  • 增强版getopt比较好用,执行命令getopt -T; echo $?,如果输出4,则代表是增强版

  • 如果短选项带argument且参数可选时,argument必须紧贴选项,例如-carg而不能是-c arg

  • 如果长选项带argument且参数可选时,argument和选项之间用“=”,例如–clong=arg而不能是–clong arg

getopt.sh
#!/bin/bash

echo original parameters=[$@]

#-o或--options选项后面是可接受的短选项,如ab:c::,表示可接受的短选项为-a -b -c,
#其中-a选项不接参数,-b选项后必须接参数,-c选项的参数为可选的
#-l或--long选项后面是可接受的长选项,用逗号分开,冒号的意义同短选项。
#-n选项后接选项解析错误时提示的脚本名字
ARGS=`getopt -o ab:c:: --long along,blong:,clong:: -n "$0" -- "$@"`
if [ $? != 0 ]; then
    echo "Terminating..."
    exit 1
fi

echo ARGS=[$ARGS]
#将规范化后的命令行参数分配至位置参数($1,$2,...)
eval set -- "${ARGS}"
echo formatted parameters=[$@]

while true
do
    case "$1" in
        -a|--along)
            echo "Option a";
            shift
            ;;
        -b|--blong)
            echo "Option b, argument $2";
            shift 2
            ;;
        -c|--clong)
            case "$2" in
                "")
                    echo "Option c, no argument";
                    shift 2
                    ;;
                *)
                    echo "Option c, argument $2";
                    shift 2;
                    ;;
            esac
            ;;
        --)
            shift
            break
            ;;
        *)
            echo "Internal error!"
            exit 1
            ;;
    esac
done

#处理剩余的参数
echo remaining parameters=[$@]
echo \$1=[$1]
echo \$2=[$2]