【发布时间】:2014-03-08 20:49:49
【问题描述】:
我正在使用标准的 getopts 逻辑。但我想知道如何使我提供的选项相互排斥。 例如
shell.sh -a SID
<accepted>
shell.sh -b SID
<accepted>
shell.sh -ab SID
Message- using ab together is the same as running shell.sh without any options supplying just SID . Help usage < ya da ya >
shell.sh
Please enter SID at the minimum. Usage < ya da ya >
shell.sh SID
<accepted>
我正在尝试使用以下内容开发此逻辑
while getopts ":a:b:" opt; do
case $opt in
a ) SID="$OPTARG";;
set var=1
b ) SID="$OPTARG";;
set var=2
\?) echo "Invalid option: -"$OPTARG"" >&2
exit 1;;
) echo "Option -"$OPTARG" requires an argument." >&2
exit 1;;
esac
done
If (( val == 1 )) then ; # option a is invoked SID supplied
<stuff>
elif (( val == 2 )) then ; # option b is invoked SID supplied
<stuff>
else # SID supplied but neither a or b is invoked
<stuff>
fi
如何强制执行互斥标志。我确信有更多的杂技演员方法可以做到这一点。我想我在这里遗漏了一些常识 - 并试图弄清楚这一点。 谢谢
$ more opt.ksh
die () {
echo "ERROR: $*. Aborting." >&2
return 1
}
var=
while getopts ":a:b:" opt; do
case $opt in
a ) SID="$OPTARG"
[ "$var" = 2 ] && die "Cannot specify option a after specifying option b"
[ "$OPTARG" = b ] && die "Do not specify b as a value for option a"
var=1
;;
b ) SID="$OPTARG"
[ "$var" = 1 ] && die "Cannot specify option b after specifying option a"
[ "$OPTARG" = a ] && die "Do not specify a as a value for option b"
var=2
;;
:) die "Must supply an argument to $OPTARG"
;;
\?) die "Invalid option: -$OPTARG. Abort"
;;
esac
done
shift $(($OPTIND - 1))
[ "$SID" ] || SID=$1
[ "$SID" ] || die "You must, at the minimum, supply SID"
我正在使用 ksh
$ ps -p $$
PID TTY TIME CMD
1261 pts/45 00:00:00 ksh
我第一次运行它。
$ . opt.ksh -a 123 -b 123 # c0
ERROR: Cannot specify option b after specifying option a. Aborting.
-bash: /home/d1ecom1/gin1: No such file or directory
$ . opt.ksh -ab 123 # c1 should reject "cant use a and b togather. try with a or b"
-nologin: .[25]: shift: 4: bad number
$ . opt.ksh -a -b # c2 same as c1's message
$ . opt.ksh -a -b 123 # c3 same as c1
$ . opt.ksh -a -b 123 # c5
$ . opt.ksh -ab 123 # c6
$ . opt.ksh -a 123 -b 123 # c7
以上所有情况 C0:C7 都应该拒绝。注意 C0 和 C7 是一样的。然而,尽管这个 C0 给出了预期的错误,而 C7 不会给出任何错误?奇怪
只有工作的人应该是
. opt.ksh -a 123
. opt.ksh -b 123
. opt.ksh 123
@hvd :TYSM 为您的回复。 我想添加一个附加标志 -p 它将提供“路径覆盖”选项。 所以也许我们必须扩展 getopt 以获取像这样的参数
die () {
echo "ERROR: $*. Aborting." >&2
exit 1
}
var=
opta=false
optb=false
while getopts ":ab:p" opt; do
case $opt in
a ) $optb && die "Cannot specify option a after specifying option b"
opta=true
;;
b ) $opta && die "Cannot specify option b after specifying option a"
optb=true
;;
p ) [ -d "$mypath" ] && die "path is invalid"
;;
\?) die "Invalid option: -$OPTARG. Abort"
;;
esac
done
shift $(($OPTIND - 1))
test $# -eq 0 && die "You must supply SID"
test $# -eq 1 || die "Too many command-line arguments"
# SID=$1
以上是旧方法。现在我有 2 个输入参数。一个是 SID,另一个是路径。是像上面那样简单还是我需要添加更多检查以防止其他不需要的组合。我想我试图解决的问题是,需要做出更多的规定来允许这个 -p 参数,它是一个可选的覆盖参数。 -p 可以与上面的任何参数共存,但我想一个要求是它应该紧跟在 -p 标志之后 所以这不应该允许 - 因为它不清楚。
shell.sh -b -p 123 /home/yadaya
再次感谢
【问题讨论】:
-
您为什么将 SID 视为选项参数?如果您只是创建不带任何参数的
-a和-b选项(通过删除:),并始终检查是否提供了非选项作为SID,这已经按预期工作了。跨度> -
用这些结果更新了帖子。还是卡住了
-
你确实需要你现在变成 cmets 的最后一部分的第一和第三行 (
shift $(($OPTIND - 1))/[ "$SID" ] || die "You must, at the minimum, supply SID"),以便单独拒绝-a。我会在短时间内将我的方法扩展为答案。 -
我实际上想添加一个额外的 -p /path 标志。我更新了我的 Q。只要你有机会。Thx
-
这可能是您应该提出的新问题,而不是对旧问题的编辑。但是,与您的
-a/-b不同,在我看来它确实可以作为选项参数(:abp:,而不是:ab:p)。在您的p)案例中,只需设置mypath=$OPTARG(并根据需要进行验证)。其余检查将保持不变,因为您仍然只有一个非选项参数。然而,虽然-p /path -a 123似乎显然有效并且会被允许,但-a -p /path 123也会如此,但-a 123 -p /path会被拒绝。你可以决定这对你是否有意义。
标签: shell awk case command-line-arguments getopts