GNU argp:如何解析只有长名称的选项?
GNU argp: How to parse option with only long name?
我想在我的 C 程序中使用 argp
来解析命令行选项。
一个要求是选项没有短名称(如 -f bar
),只有长名称(如 --foo=bar
)。
到目前为止,我的方法是将 argp_option
结构的 key
字段设置为 0
以使其不显示短名称。
我已经详细查看了 the provided examples 和 argp.h
,但我找不到一种方法来解析一个选项,该选项在提供给 [= 的解析器函数中只有一个长名称11=].
我确实发现,理论上,我可以在解析器中使用
case ARGP_KEY_ARG:
printf("%s\n", arg);
找到长选项的值(例如,当用--foo=bar
调用时,bar
会在这里打印)。但是,这似乎不是正确的方法,因为我没有看到一种简单的方法来实际判断该值属于哪个选项。这也显示了实际的命令行参数(不是选项值)。
如能提供有关我需要查看的位置的任何提示,我将不胜感激。干杯。
来自 "The GNU C Library: Argp Option Vectors — Specifying Options in an Argp Parser":
int key
The integer key
provided by the current option to the option parser.
If key
has a value that is a printable ASCII character (i.e.,
isascii (key)
is true), it also specifies a short option -char
,
where char
is the ASCII character with the code key
.
换句话说,选项的 key
字段可以是任何 int
值,如果 isascii(key)
不为零,那么它也指定了一个短选项——这意味着你可以使用非 ASCII 值(0x00
..0x7F
范围之外的值)来避免短选项。尽管不是短选项,key
的值仍然用作关联长选项的值(例如 --foo
),因此您可以像处理任何 key/short 一样处理它选项。
O/T:我在 enum
中收集了我所有的选项键作为常量,所以我不知道 0x100
在 switch
中代表什么选项,例如对于 GNU tar
,它的压缩选项可能是这样的:
enum compress_options {
// Archive file extension determines compression program
COMP_FILTER_AUTO = 'a',
// Option arg is program used to deal with compression
COMP_FILTER_ARG = 'I',
COMP_FILTER_BZIP2 = 'j',
COMP_FILTER_XZ = 'J',
COMP_FILTER_GZIP = 'z',
COMP_FILTER_COMPRESS = 'Z',
COMP_FILTER_LZIP = 0x100,
COMP_FILTER_LZMA,
COMP_FILTER_LZOP,
// Do not use archive suffix to determine compression program.
COMP_FILTER_NOAUTO,
};
那么您只需要确保下一组选项使用 0x200
,然后是 0x300
,等等,以避免选项具有相同值的问题。如果需要,您还可以使用 0x100
、0x180
、0x200
、0x280
等(例如 0x100
可能表示子命令,而 0x180
可能是该子命令的第一个选项)。
将空头期权和非空头期权分开分组很重要。 enum
中的隐式赋值取决于最后一个显式赋值的值。如果我在 COMP_FILTER_AUTO
之后立即放置 COMP_FILTER_NOAUTO
,--no-auto-compress
多头选项将有一个关联的空头选项 -b
,但实际上并没有。
虽然 是正确的,但缺少具体示例。
这里我们创建两个参数,一个可以正常使用(-b
),一个只能通过长名称使用(--foo
)。
#define ARGUMENT_BAR_SHORT 'b'
#define ARGUMENT_FOO_SHORT 0x80
static struct argp_option options[] =
{
{ "bar", ARGUMENT_BAR_SHORT, "BAR_VAL", 0, "The bar value" },
{ "foo", ARGUMENT_FOO_SHORT, "FOO_VAL", 0, "The foo value" },
}
然后在这里解析:
error_t parse_opt (int option, char* arg, struct argp_state* state)
{
switch (option)
{
case ARGUMENT_BAR_SHORT:
printf("got bar: %s", arg);
break;
case ARGUMENT_FOO_SHORT:
printf("got foo: %s", arg);
break;
}
}
对于尚未阅读其他答案的任何人:只需使用正常字符范围之外的值,0x80
已经是。如果您需要更多参数,只需增加此值即可。我也喜欢另一个答案中使用枚举的解决方案。
我还没有编译这个 ;)
我想在我的 C 程序中使用 argp
来解析命令行选项。
一个要求是选项没有短名称(如 -f bar
),只有长名称(如 --foo=bar
)。
到目前为止,我的方法是将 argp_option
结构的 key
字段设置为 0
以使其不显示短名称。
我已经详细查看了 the provided examples 和 argp.h
,但我找不到一种方法来解析一个选项,该选项在提供给 [= 的解析器函数中只有一个长名称11=].
我确实发现,理论上,我可以在解析器中使用
case ARGP_KEY_ARG:
printf("%s\n", arg);
找到长选项的值(例如,当用--foo=bar
调用时,bar
会在这里打印)。但是,这似乎不是正确的方法,因为我没有看到一种简单的方法来实际判断该值属于哪个选项。这也显示了实际的命令行参数(不是选项值)。
如能提供有关我需要查看的位置的任何提示,我将不胜感激。干杯。
来自 "The GNU C Library: Argp Option Vectors — Specifying Options in an Argp Parser":
int key
The integer
key
provided by the current option to the option parser. Ifkey
has a value that is a printable ASCII character (i.e.,isascii (key)
is true), it also specifies a short option-char
, wherechar
is the ASCII character with the codekey
.
换句话说,选项的 key
字段可以是任何 int
值,如果 isascii(key)
不为零,那么它也指定了一个短选项——这意味着你可以使用非 ASCII 值(0x00
..0x7F
范围之外的值)来避免短选项。尽管不是短选项,key
的值仍然用作关联长选项的值(例如 --foo
),因此您可以像处理任何 key/short 一样处理它选项。
O/T:我在 enum
中收集了我所有的选项键作为常量,所以我不知道 0x100
在 switch
中代表什么选项,例如对于 GNU tar
,它的压缩选项可能是这样的:
enum compress_options {
// Archive file extension determines compression program
COMP_FILTER_AUTO = 'a',
// Option arg is program used to deal with compression
COMP_FILTER_ARG = 'I',
COMP_FILTER_BZIP2 = 'j',
COMP_FILTER_XZ = 'J',
COMP_FILTER_GZIP = 'z',
COMP_FILTER_COMPRESS = 'Z',
COMP_FILTER_LZIP = 0x100,
COMP_FILTER_LZMA,
COMP_FILTER_LZOP,
// Do not use archive suffix to determine compression program.
COMP_FILTER_NOAUTO,
};
那么您只需要确保下一组选项使用 0x200
,然后是 0x300
,等等,以避免选项具有相同值的问题。如果需要,您还可以使用 0x100
、0x180
、0x200
、0x280
等(例如 0x100
可能表示子命令,而 0x180
可能是该子命令的第一个选项)。
将空头期权和非空头期权分开分组很重要。 enum
中的隐式赋值取决于最后一个显式赋值的值。如果我在 COMP_FILTER_AUTO
之后立即放置 COMP_FILTER_NOAUTO
,--no-auto-compress
多头选项将有一个关联的空头选项 -b
,但实际上并没有。
虽然
这里我们创建两个参数,一个可以正常使用(-b
),一个只能通过长名称使用(--foo
)。
#define ARGUMENT_BAR_SHORT 'b'
#define ARGUMENT_FOO_SHORT 0x80
static struct argp_option options[] =
{
{ "bar", ARGUMENT_BAR_SHORT, "BAR_VAL", 0, "The bar value" },
{ "foo", ARGUMENT_FOO_SHORT, "FOO_VAL", 0, "The foo value" },
}
然后在这里解析:
error_t parse_opt (int option, char* arg, struct argp_state* state)
{
switch (option)
{
case ARGUMENT_BAR_SHORT:
printf("got bar: %s", arg);
break;
case ARGUMENT_FOO_SHORT:
printf("got foo: %s", arg);
break;
}
}
对于尚未阅读其他答案的任何人:只需使用正常字符范围之外的值,0x80
已经是。如果您需要更多参数,只需增加此值即可。我也喜欢另一个答案中使用枚举的解决方案。
我还没有编译这个 ;)