8wDlpd.png
8wDFp9.png
8wDEOx.png
8wDMfH.png
8wDKte.png

如何在 bash 中多次调用 getopts

RaytheonXie-MSFT 1月前

19 0

我有一个通用库,用于在多个脚本中解析命令行选项,但是我也希望我的个人脚本也能够处理参数...例如common.sh:func...

我有一个通用库,用于从多个脚本解析命令行选项,但我还希望我的各个脚本也能够处理参数...例如

通用.sh:

function get_options {
    echo -e "in getoptions"
    echo $OPTIND
    while getopts ":ab:" optionName; do
       [ ... processing code ... ]
    done
}

. ./common.sh

function get_local_options {
    echo -e "in getoptions"
    echo $OPTIND
    while getopts ":xy:" optionName; do
       [ ... processing code ... ]
    done
}

get_local_options $*
OPTIND=1
get_options $*

问题是,如果我用以下命令调用 a.sh:

a.sh -x -y foo -a -b bar

get_options 在 \'foo\' 处停止处理,因为它在第一个 \'非选项\' 处停止

有什么办法可以解决这个问题而不用自己重写吗?

帖子版权声明 1、本帖标题:如何在 bash 中多次调用 getopts
    本站网址:http://xjnalaquan.com/
2、本网站的资源部分来源于网络,如有侵权,请联系站长进行删除处理。
3、会员发帖仅代表会员个人观点,并不代表本站赞同其观点和对其真实性负责。
4、本站一律禁止以任何方式发布或转载任何违法的相关信息,访客发现请向站长举报
5、站长邮箱:yeweds@126.com 除非注明,本帖由RaytheonXie-MSFT在本站《shell》版块原创发布, 转载请注明出处!
最新回复 (0)
  • foo() {
      unset OPTIND
      while getopts ...
      do
      done
    }
    
  • 只需让您的通用脚本提供用于处理这些选项和函数的列表,然后让其他脚本将它们可以处理的任何选项添加到选项列表中:

    通用脚本

    OPTS="ab:hv"
    
    common_usage() {
      echo "Common options:"
      echo "  -a     ..."
      echo "  -b arg ..."
      echo "  -h     Show usage"
      echo "  -v     Be verbose"
    }
    
    handle_common_opts() {
      case "$1" in
        a) handle_opt_a ;;
        b) handle_opt_b "$2" ;;
        h) usage; exit 0 ;;
        v) be_verbose ;;
      esac
    }
    

    . common.sh
    
    OPTS="${OPTS}xy:"
    
    usage() {
      echo "Usage: $PROG [options] [arguments...]"
      echo
      common_usage
      echo
      echo "Specific options:"
      echo "  -x     ..."
      echo "  -y arg ..."
    }
    
    while getopts "$OPTS" OPT; do
      handle_common_opts "$OPT" "$OPTARG"
      case "$OPT" in
        x) handle_opt_x ;;
        y) handle_opt_y "$OPTARG" ;;
        \?) usage; exit 1 ;;
      esac
    done
    shift $((OPTIND-1))
    
    handle_remaining_args "$@"
    

    这样做有以下优点:不需要对 的内部管理进行任何改动 getopts ,用户可以以任何顺序指定常用和特定的选项,它可以在一个地方处理无法识别的选项,并且使使用功能更易于维护。

  • Ndx 1月前 0 只看Ta
    引用 4

    我设法让它工作,不确定这是否是你想要的:

    $ cat common.sh
    function get_options {
        while getopts ":ab:" optionName
        do
            echo "get_options: OPTARG: $OPTARG, optionName: $optionName"
        done
    }
    $ cat a.sh
    #!/bin/bash
    
    . ./common.sh
    
    function get_local_options {
        while getopts ":xy:" optionName; do
            case $optionName in
                x|y) 
                    echo "get_local_options: OPTARG: $OPTARG, optionName: $optionName"
                    last=$OPTIND;;
                *) echo "get_local_options, done with $optionName"
                    break;;
            esac;
        done
    }
    
    last=1
    get_local_options $*
    shift $(($last - 1))
    OPTIND=1
    get_options $*
    $ ./a.sh -x -y foo -a -b bar
    get_local_options: OPTARG: , optionName: x
    get_local_options: OPTARG: foo, optionName: y
    get_local_options, done with ?
    get_options: OPTARG: , optionName: a
    get_options: OPTARG: bar, optionName: b
    $ ./a.sh -a -b bar
    get_local_options, done with ?
    get_options: OPTARG: , optionName: a
    get_options: OPTARG: bar, optionName: b
    
  • 我认为您需要让第一个 getopts 处理器构建一个它未处理的命令行元素列表。此外,它必须忽略 getopts 调用的返回值:getopts 可能返回非零值,因为它正在处理它不知道如何处理的命令行开关的参数。因此,第一个 getopts 处理器需要转到命令行的末尾。您不能相信它自己对选项何时停止以及参数何时开始的判断。(好吧,一旦您连续获得两个参数,您大概就可以跳过其余的参数,但我将把它留作练习。)

    那么就像这样:

    #!/bin/bash
    
    UNHANDLED=()
    
    function getxy {
      while ((OPTIND<=$#)); do
        if getopts ":xy:" opt; then
          case $opt in
            x|y) echo "getxy opt=$<$opt> OPTARG=<$OPTARG>";;
            *) UNHANDLED+=(-$OPTARG);;
          esac
        else
            UNHANDLED+=(${!OPTIND})
            let OPTIND++
        fi
      done
    }
    
    function getab {
      while getopts ":ab:" opt; do
      case $opt in
        a|b) echo "getab opt=$<$opt> OPTARG=<$OPTARG>";;
        *) echo "getab * opt=<$opt> OPTARG=<$OPTARG>";;
      esac
      done
    }
    
    echo "--- getxy ---"
    OPTIND=1
    getxy "$@"
    # now we reset OPTIND and parse again using the UNHANDLED array
    echo "--- getab ---"
    OPTIND=1
    set -- "${UNHANDLED[@]}"
    getab "$@"
    # now we get remaining args
    shift $((OPTIND-1))
    for arg; do
        echo "arg=<$arg>"
    done
    
  • 这里有一些不错的答案,但我认为有一个简单的解决方案:您需要增加它 OPTIND 而不是将其重置为 1。即

    . ./common.sh
    
    function get_local_options {
        echo -e "in getoptions"
        echo $OPTIND
        while getopts ":xy:" optionName; do
           [ ... processing code ... ]
        done
    }
    
    get_local_options $*
    (( OPTIND++ ))
    get_options $*
    

    这样,您将跳过 foo 参数列表中的单词。

返回
作者最近主题: