为什么我不能使用getopt在选项和可选参数之间有空格?

时间:2014-10-17 19:00:00

标签: bash getopt

使用getopt解析命令行参数时,可以在选项标志和必需参数的参数之间放置一个空格,但不在可选参数之间。只有在选项之后才会解析可选参数。

TEMP=`getopt -o p:q:: -n 'mkqueue.sh' -- "$@"`

if [ $? != 0 ] ; then echo "Terminating..." >&2 ; exit 1 ; fi

# Note the quotes around `$TEMP': they are essential!
eval set -- "$TEMP"

# Now go through all the options
while true ; do
    case "$1" in
        -p) echo "Option p, argument \`$2'" ; shift 2 ;;
        -q) 
            case "$2" in
                "") echo "Option q, no argument"; shift 2 ;;
                *)  echo "Option q, argument \`$2'" ; shift 2 ;;
            esac ;;
        --) shift ; break ;;
        *) echo "Internal error!" ; exit 1 ;;
    esac
done

(基于此:http://software.frodo.looijaard.name/getopt/docs/getopt-parse.bash

当我运行没有可选参数的脚本时,它可以工作:

./mkqueue.sh -p adfsa -q
Option p, argument `adfsa'
Option q, no argument

如果我尝试将可选参数添加到-q,并且中间有空格,则它不起作用:

./mkqueue.sh -p adfsa -q sdfasdfa
Option p, argument `adfsa'
Option q, no argument

如果你在选项和参数之间没有空格,即使必需的参数与空格一起使用,它也可以工作:

./mkqueue.sh -p adfsa -qsdfasdfa
Option p, argument `adfsa'
Option q, argument `sdfasdfa'

这是否有解决方法?

1 个答案:

答案 0 :(得分:8)

该限制记录在getopt

的联机帮助页中
  

简单的短选项是` - '接着是一个短选项字符。如果该选项具有必需参数,则可以直接在选项字符之后或作为下一个参数(即由命令行上的空格分隔)写入。 如果该选项具有可选参数,则必须在选项字符后直接写入(如果存在)。

尽管getopt允许"可选选项参数",但它们通常被认为是个坏主意。例如,Posix Utility Syntax Guidelines包括:

  

准则7 :选项参数不应是可选的。

本指南的基本原因是,按照惯例,命令行选项的参数可以是任何字符串。例如,它可能看起来像另一种选择。它可能是文字字符串--,通常表示选项列表的结尾。它可能是一个空字符串。任何东西。

"可选选项参数的问题"知道不提供可选选项参数的唯一方法是将以下参数识别为选项或--。但这意味着可选参数的值不能以-开头。

或者,您可以选择getopt命令行实用程序采用的解决方案,即要求可选参数紧跟在该选项之后,而没有插入空格(即,在同一个单词中) 。但在这种情况下,可选参数不能是空字符串。

不是创建复杂的规则来解决这些歧义,而是由Posix推荐的简单解决方案(并且,我的权限很少) - 是不允许选项参数是可选的。如果命令行选项有一个共同的值,并且您希望通过不要求输入来简化用户的生命周期,那么实现两个不同的命令行选项,其中一个选择参数,另一个选择参数没有按'吨。常用技术包括对不带参数的版本使用小写字符,对带有参数的版本使用相应的大写字符,或者对该版本使用长选项(--option)。

对于那些喜欢密集和精心解释的人,Posix rationale解释说:

  

准则7允许任何字符串作为选项参数; option-argument可以以任何字符开头,可以是---,也可以是空字符串。例如,命令pr -h -pr -h --pr -h -dpr -h +2pr -h ''包含选项参数-,{{1}分别是},---d和空字符串。相反,命令+2将-d视为一个选项,而不是参数,因为pr -h -- -d在这里是一个选项参数,而不是分隔符。