c++ - 如何使用 boost::program_options 创建选项别名?

标签 c++ c++11 boost boost-program-options

我希望能够使用 boost::program_options 创建选项别名,将其参数存储在同一键/标签下。

我的软件架构根据值 argv[1] 使用不同的专用选项解析器。不过,有些选项是共享的,例如我的选项 --inputs

inputOptions.add_options()
        ("--inputs",
         po::value< std::vector<std::string> >()->value_name("paths"),
         "List of files to edit.\n");

为了与旧版本的程序兼容,我想向其中一个子解析器添加一个兼容性选项--input,该选项将其参数存储在“--inputs”下。理想情况下,该选项最多应采用一个参数,而不是任意多个。但是,如果您提供一个使 --input--inputs 相同的解决方案,我想它也很好,因为在这种情况下,位置选项被发送到“--inputs” “无论如何。

感谢您的帮助!

最佳答案

您可以使用extra_parser(请参阅文档中的Non-conventional syntax),即可用于在进一步处理输入之前操作输入 token 的解析器。它可用于将 --run 翻译为 --command=run 等。

额外的解析器是一个具有以下签名的仿函数:

std::pair<std::string, std::string>(const std::string &s)

它应该在 pair::first 中返回选项名称,并在 pair::second 中返回(可选)选项值。空的 pair::first 意味着额外的解析器还没有解析任何东西。值(即pair::second)可以为空 - 仅解析选项名称。如果返回的对有效,则使用名称或名称/值对,而不是通过正常机制解析原始 token 。

首先,我们编写别名函数:

using OptionAliases = std::map<std::string, std::string>;

std::pair<std::string, std::string>
renameOptions(const std::string &token, const OptionAliases &aliases)
{
    auto rtoken(boost::make_iterator_range(token));

    // consume "--" prefix
    if (!boost::algorithm::starts_with(rtoken, "--")) { return { "", "" }; }
    rtoken.advance_begin(2);

    // find equal sign (returns iterator range)
    const auto eq(boost::algorithm::find_first(rtoken, "="));

    // extract option (between "--prefix" and "="/end()) and map it to output
    const auto faliases(aliases.find(std::string(rtoken.begin(), eq.begin())));
    if (faliases == aliases.end()) { return { "", "" }; }

    // return remapped option and (optionally) value after "="
    return std::make_pair(faliases->second
                          , std::string(eq.end(), rtoken.end()));
}

它只是将输入标记拆分为 --name=value(如果存在则无值)没有 = 符号),如果在提供的别名映射中找到该名称,则返回 (remapped-name, value)

然后,我们使用 lambda 创建解析器本身:

boost::program_options::ext_parser optionAlias(OptionAliases &&aliases)
{
    return [aliases{std::move(aliases)}](const std::string &token)
    {
        return renameOptions(token, aliases);
    };
}

(至少需要 C++14,对于 C++11 更改为 return [aliases](con...)

您可以将此解析器插入到cmdline解析器中:

parser.extra_parser(optionAlias({{"mark.twain", "samuel.clemens"}
                                  , {"lewis.caroll", "charles.dodgson"}}));

现在,在上面的示例中,--mark.twain--samuel.clemens 都将指向 vars["samuel.clemens"] --lewis.caroll--charles.dodgson 都将指向 vars["charles.dodgson"]

注意事项:

  • 仅适用于命令行解析器。
  • 需要 allow_long 样式(带有 -- 前缀的长选项)。可以在代码中更改。
  • 需要 long_allow_adjacent 样式(使用 = 在一个 token 中允许的值)。也可以在代码中进行更改。
  • 如果有任何解析为 --alias 的非选项标记,那么它也会被翻译,因为没有上下文。没有办法规避。

    示例:如果有一个名为 name 的选项需要值并且使用 long_allow_next 样式,则 --name=--mark.twain 将被解析为选项 name ,其值为 --mark.twain (如预期),而 --name --mark.twain 将被解析为带有值 samuel.clemens 的选项 name

    除此之外,它会按预期工作。

希望有帮助。

关于c++ - 如何使用 boost::program_options 创建选项别名?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42065383/

相关文章:

c++ - C++0x 是否支持 std::wstring 与 UTF-8 字节序列的转换?

c++ - 如何通过预处理文件来访问 O(1) 中文件中的一行?

c++ - 在黑莓 10 中访问单例类方法

c++ - 点云库 - 如何将单个 RGB 值分配给整个点云?

c++ - 创建一个接口(interface)来模拟 C++ 对象

c++ - std 或 boost 中是否有类似循环预付的东西?

c++ - 在 Windows 运行时组件中使用 WRL 获取文件夹路径返回空字符串

c++ - 为什么 clang 并不总是为相同的静态 constexpr 产生恒定值

c++ - 摆脱 ATLTRACE 输出中显示的 atlTraceGeneral 类别

c++ - 在什么情况下 C++ 会在编译时进行数组边界检查?