3

我正在使用伟大的fmt C++ 库来更优雅地格式化字符串。

我想将一个非变量参数列表传递给fmt::format. 它可以是std::vector, 或std::string, 或其他任何东西,但它始终与格式字符串匹配。

所以fmt::format像:

std::string message = fmt::format("The answer is {} so don't {}", "42", "PANIC!");

但我想要的是:

std::vector<std::string> arr;
arr.push_back("42");
arr.push_back("PANIC!");
std::string message = fmt::format("The answer is {} so don't {}", arr);

有没有办法/解决方法这样做?

4

3 回答 3

7

您可以vformat从您的vector. 这似乎有效:

std::string format_vector(std::string_view format,
    std::vector<std::string> const& args)
{
    using ctx = fmt::format_context;
    std::vector<fmt::basic_format_arg<ctx>> fmt_args;
    for (auto const& a : args) {
        fmt_args.push_back(
            fmt::internal::make_arg<ctx>(a));
    }

    return fmt::vformat(format,
        fmt::basic_format_args<ctx>(
            fmt_args.data(), fmt_args.size()));
}

std::vector<std::string> args = {"42", "PANIC!"};
std::string message = format_vector("The answer is {} so don't {}", args);
于 2020-01-15T03:26:55.147 回答
4

添加一个额外的图层,例如:

template <std::size_t ... Is>
std::string my_format(const std::string& format,
                      const std::vector<std::string>& v,
                      std::index_sequence<Is...>)
{
    return fmt::format(format, v[Is]...);
}


template <std::size_t N>
std::string my_format(const std::string& format,
                      const std::vector<std::string>& v)
{
    return my_format(format, v, std::make_index_sequence<N>());
}

用法是:

std::vector<std::string> arr = {"42", "PANIC!"};
my_format<2>("The answer is {} so don't {}", arr);

您可能会在operator ""_format编译时获得有关预期大小的信息

于 2018-02-20T03:32:29.447 回答
1

如果不对 fmt 库进行更改,这似乎是不可能的。 使用表示多个参数的or对象的fmt::format调用,但提供创建or对象的唯一方法是通过另一个可变参数函数,这意味着必须在编译时知道参数的数量和类型。fmt::vformatfmt::format_argsfmt::wformat_argsformat_argswformat_args

因此,您可以编写一个包装器来解包 a std::tupleorstd::array并将其元素传递给fmt::format,因为其中元素的数量和类型在编译时是已知的。但是你不能对 a std::vectorstd::list等做同样的事情,因为这些容器的大小在运行时会有所不同。

于 2018-02-20T00:49:54.007 回答