如何使用 `fmt::format` 使 `boost::stacktrace` 可格式化

How to make `boost::stacktrace` formattable with `fmt::format`

我正在努力让 boost 的堆栈跟踪库与 fmt 互操作,问题似乎是我不能完全专业化 boost::stacktrace::basic_stacktrace。有人知道根本问题是什么吗?

#include <fmt/core.h>
#include <boost/stacktrace.hpp>

using BoostTraceStackFrame = std::remove_reference_t<decltype(boost::stacktrace::stacktrace().as_vector().front())>;
template <> struct fmt::formatter<BoostTraceStackFrame> : formatter<string_view>
{
    template <typename FORMAT_CONTEXT> auto format(const BoostTraceStackFrame &rhs, FORMAT_CONTEXT &ctx)
    {
        return fmt::format(ctx.out(), "{}:{}\n", rhs.source_file(), rhs.source_line());
    }
};

using BoostTraceType = std::remove_reference_t<decltype(boost::stacktrace::stacktrace())>;
template <> struct fmt::formatter<BoostTraceType> : formatter<string_view>
{
    //possibly BoostTraceType = boost::stacktrace::basic_stacktrace ?
    template <typename FORMAT_CONTEXT> auto format(const BoostTraceType &rhs, FORMAT_CONTEXT &ctx)
    {
        return fmt::format(ctx.out(), "{}", fmt::join(rhs.as_vector(), "\n"));
    }
};

int main()
{
 auto trace = boost::stacktrace::stacktrace();
 fmt::print("Foo trace {}",trace);
}

On godbolt

三件小事:

  1. Missing/wrongheaders。您需要包含 fmt/ranges.h 以支持格式范围。

  2. 错误的重载。格式化成迭代器的函数是fmt::format_to,不是fmt::format.

  3. 这些:

using BoostTraceStackFrame = std::remove_reference_t<decltype(boost::stacktrace::stacktrace().as_vector().front())>;
using BoostTraceType = std::remove_reference_t<decltype(boost::stacktrace::stacktrace())>;

相当long-winded的写作方式:

using BoostTraceStackFrame = boost::stacktrace::frame;
using BoostTraceType = boost::stacktrace::stacktrace;

特别是,问题是您写第一个的方式是 boost::stacktrace::frame const 而不需要是 const.