如何使用 fmt 库格式化指针?

How to format pointers using fmt library?

我正在听取一些人的建议并研究 fmt 库: http://fmtlib.net

它似乎具有我需要的功能,并声称支持 %p(指针),但是在编译我使用 %p 的代码时,我得到一长串模板错误(无法理解)。我会post他们在这结束时。

如果我取出 %p 和相应的指针参数,那么它会在 VS2017 c++17 上编译。

但是,我不知道如何解码模板错误,或者如何深入了解为什么它首先不接受 %p 参数。

我试过将参数转换为 (void*) - 同样的问题。
我尝试在格式化程序 {} 中使用 python 样式语法 - 同样的问题。
我已将 %p 位与其余格式分开 - 同样的问题。

我看到有对用户类型的支持 - 但在这种情况下,我只想将其输出为原始指针值。我可以跳过它,毕竟指针地址有多大价值,真的吗?但这当然意味着在从 sprintffmt::format 的转换过程中需要做更多的工作来寻找所有 %p 和 "do something with them",例如删除它们。

但文档似乎表明支持 %p - http://fmtlib.net/latest/syntax.html(大约向下 3/4 - 在 'pointer' 或 'p' 上搜索)。

这是调用函数:(注意:pAccels 被声明为 const ACCEL *

    m_hAccel = ::CreateAcceleratorTable(const_cast<LPACCEL>(pAccels), (int)count);
    if (!m_hAccel)
    {
        auto error = GetLastError();
        throw CWinAPIErrorException(__FUNCTION__, "CreateAcceleratorTable", fmt::format("%p,%u", pAccels, count), error);
    }

这是诊断信息:

1>c:\users\steve\source\fmt\include\fmt\core.h(1073): error C2825: 'fmt::v5::internal::get_type<Context,Arg>::value_type': must be a class or namespace when followed by '::'
1>        with
1>        [
1>            Context=fmt::v5::basic_format_context<std::back_insert_iterator<fmt::v5::internal::buffer<char>>,char>,
1>            Arg=const ACCEL *
1>        ]
1>c:\users\steve\source\fmt\include\fmt\core.h(1081): note: see reference to class template instantiation 'fmt::v5::internal::get_type<Context,Arg>' being compiled
1>        with
1>        [
1>            Context=fmt::v5::basic_format_context<std::back_insert_iterator<fmt::v5::internal::buffer<char>>,char>,
1>            Arg=const ACCEL *
1>        ]
1>c:\users\steve\source\fmt\include\fmt\core.h(1190): note: see reference to function template instantiation 'unsigned __int64 fmt::v5::internal::get_types<Context,const ACCEL*,size_t>(void)' being compiled
1>        with
1>        [
1>            Context=fmt::v5::basic_format_context<std::back_insert_iterator<fmt::v5::internal::buffer<char>>,char>
1>        ]
1>c:\users\steve\source\fmt\include\fmt\core.h(1190): note: while compiling class template member function 'unsigned __int64 fmt::v5::format_arg_store<fmt::v5::basic_format_context<std::back_insert_iterator<fmt::v5::internal::buffer<Char>>,Char>,const ACCEL *,size_t>::get_types(void)'
1>        with
1>        [
1>            Char=char
1>        ]
1>c:\users\steve\source\fmt\include\fmt\core.h(1478): note: see reference to class template instantiation 'fmt::v5::format_arg_store<fmt::v5::basic_format_context<std::back_insert_iterator<fmt::v5::internal::buffer<Char>>,Char>,const ACCEL *,size_t>' being compiled
1>        with
1>        [
1>            Char=char
1>        ]
1>c:\users\steve\source\tbx\wapi\acceleratortable.cpp(58): note: see reference to function template instantiation 'std::basic_string<char,std::char_traits<char>,std::allocator<char>> fmt::v5::format<char[6],const ACCEL*,size_t,0>(const S (&),const ACCEL *const &,const size_t &)' being compiled
1>        with
1>        [
1>            S=char [6]
1>        ]
1>c:\users\steve\source\fmt\include\fmt\core.h(1073): error C2510: 'value_type': left of '::' must be a class/struct/union
1>c:\users\steve\source\fmt\include\fmt\core.h(1073): error C2065: 'type_tag': undeclared identifier
1>c:\users\steve\source\fmt\include\fmt\core.h(1073): error C2131: expression did not evaluate to a constant
1>c:\users\steve\source\fmt\include\fmt\core.h(1073): note: a non-constant (sub-)expression was encountered
1>c:\users\steve\source\fmt\include\fmt\core.h(1197): error C2131: expression did not evaluate to a constant
1>c:\users\steve\source\fmt\include\fmt\core.h(1082): note: failure was caused by non-constant arguments or reference to a non-constant symbol
1>c:\users\steve\source\fmt\include\fmt\core.h(1082): note: see usage of 'value'

要格式化指针,您可以将其转换为 void*:

std::string s = fmt::format("{},{}", static_cast<void*>(pAccels), count);

或者用fmt::ptr包裹起来:

std::string s = fmt::format("{},{}", fmt::ptr(pAccels), count);

godbolt 的工作示例:https://godbolt.org/z/sCNbjr

请注意 format 使用 Python-like 格式字符串语法,而不是 printf 和 returns 一个 std::string 对象。