GNU argp:如何解析只有长名称的选项?

GNU argp: How to parse option with only long name?

我想在我的 C 程序中使用 argp 来解析命令行选项。

一个要求是选项没有短名称(如 -f bar),只有长名称(如 --foo=bar)。

到目前为止,我的方法是将 argp_option 结构的 key 字段设置为 0 以使其不显示短名称。

我已经详细查看了 the provided examplesargp.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 中收集了我所有的选项键作为常量,所以我不知道 0x100switch 中代表什么选项,例如对于 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,等等,以避免选项具有相同值的问题。如果需要,您还可以使用 0x1000x1800x2000x280 等(例如 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 已经是。如果您需要更多参数,只需增加此值即可。我也喜欢另一个答案中使用枚举的解决方案。


我还没有编译这个 ;)