在Linux bash中,可以用以下三种方式解析命令行参数:
-
直接处理:使用$1,$2,$3…进行解析
-
getopts: 单个字符选项的情况,例如:-n 10 -f file.txt等选项
-
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]