Bash使用单个必需参数来获取多个标志

时间:2016-05-23 17:02:42

标签: bash getopts

我无法绕过如何使getopts接受带有单个参数的多个标志。例如。

标志:儿童:a - Adam b - Beth c - Cathy d - Duncan

必需参数:冰淇淋口味

(不是我的剧本真正做的,但最容易拼写出这种方式)

我现在能做什么:

./flavor -a chocolate

./flavor -c chocolate 

我想做什么:

./flavor -acd chocolate

而不是我能做到的唯一一件事:

./flavor -a chocolate -c chocolate -d chocolate

我尝试使用getopts的每个变体要么根本看不到参数,要么在每个标志声明后都需要它。我觉得我错过了一些显而易见的东西,它会给我一个“啊哈”的时刻,但我自己并没有看到它。有人能指出我正确的方向吗?

============================================

while getopts ":a:b:c:d:" opt; do
  case $opt in
        a)
          do stuff with $OPTARG
         ;;
        b) 
          do stuff with $OPTARG
         ;;
        c)
          do stuff with $OPTARG
         ;;
        d)
          do stuff with $OPTARG
         ;;
        :)
         echo "Option -$OPTARG requires an argument." >&2
         exit 1
         ;;
  esac
done

============================================

如果这是一个非常愚蠢的问题我很抱歉,感谢您的时间。

2 个答案:

答案 0 :(得分:1)

getopts 不支持需要参数的堆叠选项。您只能堆叠不带参数的选项。

为了做你的建议,需要在参数处理中复杂化更多。基本上,您必须预先处理选项以找出非选项参数的开始位置。这不是特别漂亮或推荐,但这确实是你要求的,而不需要额外的选项:

_GETOPTS=":abcd"
while getopts "${_GETOPTS}" opt
do
  case $opt in
        abcd)
         :
         ;;
        \?)
         echo "Unsupported option (-${OPTARG})" >&2
         exit 1
         ;;
  esac
done
eval "FIRSTARG=\${${OPTIND}}"
if [ -z "${FIRSTARG}" ]
then
   echo "Flavor argument required."
   exit 1
fi
OPTIND=1
while getopts "${_GETOPTS}" opt
do
  case $opt in
        a)
         echo do a stuff with ${FIRSTARG}
         ;;
        b)
         echo do b stuff with ${FIRSTARG}
         ;;
        c)
         echo do c stuff with ${FIRSTARG}
         ;;
        d)
         echo do d stuff with ${FIRSTARG}
         ;;
  esac
done

结果:

$ ./flavor -acd chocolate
do a stuff with chocolate
do c stuff with chocolate
do d stuff with chocolate

可以在第一个非选项参数之后访问其他参数,但这需要更多的工作。

请注意,在多个 getopts 语句中处理同一组选项很容易造成长期维护问题,因为对于两个 case处理的选项可能太容易了语句不同步。

答案 1 :(得分:0)

这样做有点粗略:

 #!/bin/bash
while getopts ":abcdf:" opt; do
case $opt in
f)
FLAVOUR="$OPTARG"

if [ ${ADAM_IN:-0} -eq 1 ]
then
  echo "Adam gets $FLAVOUR"
  ADAM_IN=0
fi
if [ ${BETH_IN:-0} -eq 1 ]
then
  echo "Beth gets $FLAVOUR"
  BETH_IN=0
fi
if [ ${CATHY_IN:-0} -eq 1 ]
then
  echo "Cathy gets $FLAVOUR"
  CATHY_IN=0
fi
if [ ${DUNCAN_IN:-0} -eq 1 ]
then
  echo "Duncan gets $FLAVOUR"
  DUNCAN_IN=0
fi
;;
a)
  ADAM_IN=1
;;
b)
  BETH_IN=1
;;
c)
  CATHY_IN=1
;;
d)
  DUNCAN_IN=1
;;
:)
echo "invalid argument" 2>/dev/null
exit 1
;;
esac
done

将脚本保存为ice_cream_allocator并按以下方式运行:

$ ./ice_cream_allocator -abcf"Chocolate" -df"Vanila" -cf"Strawberry"

<强>输出

Adam gets Chocolate
Beth gets Chocolate
Cathy gets Chocolate
Duncan gets Vanila
Cathy gets Strawberry

嗯,我为Cathy设置了一个软角,所以给她两种口味;)