2

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

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

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

我已经详细查看了提供的示例argp.h但我找不到解析选项的方法,该选项在提供给的解析器函数中只有一个长名称argp

我确实发现,理论上,在我可以使用的解析器中

case ARGP_KEY_ARG:
  printf("%s\n", arg);

查找长选项的(例如,当使用 , 调用时--foo=barbar将在此处打印)。但是,这似乎不是正确的方法,因为我没有看到一种简单的方法来实际判断该值属于哪个选项。这也显示了实际的命令行参数(不是选项值)。

我会感谢有关我需要查看的任何提示。干杯。

4

2 回答 2

4

来自“GNU C 库:Argp 选项向量 — 在 Argp 解析器中指定选项”

int key

key当前选项提供给选项解析器的整数。如果key有一个值是可打印的 ASCII 字符(即为 isascii (key)真),它还指定一个短选项-char,其中char是带有代码的 ASCII 字符key

换句话说,key选项的字段可以是任何int值,如果isascii(key)不为零,那么它还指定了一个短选项——这意味着您可以使用非 ASCII 值(范围之外的值0x00.. 0x7F)来避免短选项。尽管不是短选项,但 的值key仍用作相关长选项的值(例如--foo),因此您可以像处理任何键/短选项一样处理它。

O/T:我将我所有的选项键都收集在一个enum常量中,所以我不知道0x100在 a 中代表什么选项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, then0x300等,以避免选项具有相同值的问题。如果需要,您还可以使用0x100, 0x180, 0x200,0x280等(例如,0x100可能代表一个子命令并且0x180可能是该子命令的第一个选项)。

将空头期权和非空头期权分开分组很重要。an 中的隐式赋值enum取决于最后一个显式赋值的值。如果我在COMP_FILTER_NOAUTO之后立即放置COMP_FILTER_AUTO--no-auto-compress多头期权将有一个相关的空头期权-b,但实际上并没有。

于 2017-12-09T11:47:48.060 回答
1

虽然公认的答案是正确的,但它缺少一个具体的例子。

这里我们创建了两个参数,一个可以正常使用(-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已经是。如果您需要更多参数,只需增加此值即可。我也喜欢其他答案中使用枚举的解决方案。


我也没有编译这个;)

于 2021-04-29T15:40:12.830 回答