2

我正在编写大量的 Bash 函数,这些函数能够从使用位置参数命名参数(由 提供getopts)传递的命令行获取不同的信息。这个想法是位置参数主要用于快速和直接的人工控制,而命名参数主要用于其他功能的清晰度和控制。为了说明我的想法,考虑一个可以从一件事转换为另一件事的函数。从广义上讲,可以在快速模式或高级模式中使用此功能。在快速模式中,指定的参数很少,通常是位置参数,而在高级模式中,可以指定许多参数,通常是命名参数。举些例子...

快速模式

可以通过以下方式使用此模式:

function fileName1 fileName2

它使用内部假设和自主进行的测量将一个文件转换为另一个文件。

高级模式

可以通过以下方式使用此模式:

function -i fileName1 -o fileName2 -m -r 100 -v

此模式也可以通过以下方式使用:

function -m -v fileName1 fileName2 -r 100

请注意,-v不接受任何参数。它只是一个选项(指定诸如详细程度之类的内容)。


因此,这种情况下的逻辑是假定第一个位置参数是,fileName1并且假定第二个位置参数是,fileName2 除非使用-ior-o选项指定这些文件名中的任何一个,在这种情况下使用它们是因为它们是给定的更高的优先级(忽略第一个和第二个位置参数)。

我要求以尽可能一般的方式实现这些类型的需求的建议和指导,因为这种方法将应用于包含 150 多个函数的库。

4

3 回答 3

2

请注意,如果使用getopts所有位置参数必须在任何选项之后。第二个警告是,如果您使用getopts解析函数的参数,则必须OPTIND在解析任何选项后将全局变量重置为 1。否则我认为解决方案很简单:

argparse.sh

#!/usr/bin/env bash

func() {
    local args=$*
    local input output verbose
    local r="unset" m=0 verbose=0

    while getopts i:o:mr:v opt; do
        case "$opt" in
            i) input="$OPTARG" ;;
            o) output="$OPTARG" ;;
            r) r="$OPTARG" ;;
            m) m=1 ;;
            v) verbose=1 ;;
        esac
    done
    shift $((OPTIND-1))
    OPTIND=1    # reset global variable

    [[ $input ]]  || input=$1
    [[ $output ]] || output=$2

    echo "=== func $args ==="
    echo "input: $input"
    echo "output: $output"
    echo "verbose: $verbose"
    echo "r: $r"
    echo "m: $m"
    echo
}

func fileName1 fileName2
func -i fileName1 -o fileName2 -m -r 100 -v
func -m -v -r 100 fileName1 fileName2

输出

=== func fileName1 fileName2 ===
input: fileName1
output: fileName2
verbose: 0
r: unset
m: 0

=== func -i fileName1 -o fileName2 -m -r 100 -v ===
input: fileName1
output: fileName2
verbose: 1
r: 100
m: 1

=== func -m -v -r 100 fileName1 fileName2 ===
input: fileName1
output: fileName2
verbose: 1
r: 100
m: 1
于 2013-12-24T19:02:39.080 回答
1

改编自http://rsalveti.wordpress.com/2007/04/03/bash-parsing-arguments-with-getopts/

总是先尝试 getopts 怎么样,如果失败则恢复为仅使用 $1/$2/etc。

TEST=
SERVER=
PASSWD=
VERBOSE=
while getopts “ht:r:p:v” OPTION
do
     case $OPTION in
         h)
             usage
             exit 1
             ;;
         t)
             TEST=$OPTARG
             ;;
         r)
             SERVER=$OPTARG
             ;;
         p)
             PASSWD=$OPTARG
             ;;
         v)
             VERBOSE=1
             ;;
         ?)
             usage
             exit
             ;;
     esac
done

if [[ -z $TEST ]] || [[ -z $SERVER ]] || [[ -z $PASSWD ]]
then
     // Set variables from $1/$2/etc....
fi
于 2013-12-24T18:06:29.893 回答
1

我使用@getopt@ 来解析命令行参数。我现在不记得它是否能解决你的问题。但这对我很有用。

http://linux.die.net/man/1/getopt

于 2013-12-24T17:50:56.127 回答