0

0

c++20的std::format如何处理自定义类型? (实现std::formatter)

冰火之心

冰火之心

发布时间:2026-01-17 14:33:15

|

655人浏览过

|

来源于php中文网

原创

std::formatter特化必须为const成员函数且parse()/format()分离实现;漏const或未正确解析格式说明符将导致编译失败或std::format_error。

c++20的std::format如何处理自定义类型? (实现std::formatter)

std::formatter 特化必须满足 const 限定和 parse() / format() 分离

直接在自定义类型上加 operator 或重载 std::format 不起作用——std::format 只认 std::formatter 特化。关键约束有两条:特化模板必须是 const 成员函数,且 parse()format() 必须分离实现。漏掉 const 会导致编译失败(常见错误:「no matching function for call to format」或「formatter does not satisfy formatter」)。

典型结构如下:

template <>
struct std::formatter {
    constexpr auto parse(format_parse_context& ctx) -> format_parse_context::iterator {
        // 解析格式说明符,如 "{}", "{:x}", "{:8}"
        return ctx.end();
    }
    template 
    auto format(const MyType& t, FormatContext& ctx) const -> FormatContext::iterator {
        // 写入格式化后的内容到 ctx.out()
        return format_to(ctx.out(), "MyType{{val={}}}", t.val);
    }
};
  • parse() 必须声明为 constexpr,返回 ctx.end() 表示接受所有默认格式(若不支持任何格式说明符,可直接返回)
  • format() 参数中 const MyType&const 成员函数限定缺一不可
  • 不要在 format() 中调用 std::format 递归格式化自身字段(易触发无限模板实例化),改用 format_to

处理格式说明符(如 "{:x}"、"{:8}")需手动解析 ctx.input()

format_parse_context::iterator 实际是指向格式字符串中 {} 内容的迭代器,比如 "{:04x}"ctx.begin() 指向 '0'。你得自己跳过空格、识别前缀、提取数字等——std::formatter 不提供现成解析器。

例如支持 "x"(十六进制)和宽度(如 "8"):

立即学习C++免费学习笔记(深入)”;

白瓜AI
白瓜AI

白瓜AI,一个免费图文AI创作工具,支持 AI 仿写,图文生成,敏感词检测,图片去水印等等。

下载
constexpr auto parse(format_parse_context& ctx) -> format_parse_context::iterator {
    auto it = ctx.begin();
    auto end = ctx.end();
    if (it != end && *it == 'x') {
        hex_ = true;
        ++it;
    }
    if (it != end && std::isdigit(*it)) {
        width_ = static_cast(*it - '0');
        ++it;
    }
    return it; // 必须返回解析结束位置
}
  • 未识别的字符不能忽略——必须停在第一个非法位置,否则 std::format 会报 std::format_error
  • 宽度建议存为 intstd::size_t 成员变量,供 format() 使用;不要在 format() 中重新解析
  • 不支持的说明符(如 "f")应让 parse() 报错:抛出 std::format_error("unknown format specifier")

嵌套格式化字段(如 struct 成员)要用 format_to + std::make_format_args

MyTypeint id;std::string name;,不能写 format_to(ctx.out(), "{} {}", t.id, t.name) —— 这会尝试调用 std::formatter 等,但它们不是你特化的类型,且上下文类型不匹配。

正确做法是显式构造参数包并复用底层格式设施:

template 
auto format(const MyType& t, FormatContext& ctx) const -> FormatContext::iterator {
    if (hex_) {
        return format_to(ctx.out(), "MyType{{id={:x}, name={}}}", t.id, t.name);
    } else if (width_ > 0) {
        return format_to(ctx.out(), "MyType{{id={:0{}}d, name={}}}", t.id, width_, t.name);
    }
    return format_to(ctx.out(), "MyType{{id={}, name={}}}", t.id, t.name);
}
  • format_to 是安全的,它接受任意已支持类型的参数(intstd::stringstd::string_view 等)
  • 注意 {:0{}}d 这种嵌套占位符:第二个 {} 会从参数列表取 width_ 值,用于动态指定宽度
  • 所有字段类型必须本身支持 std::formatter(内置类型和标准容器基本都支持,自定义类型则需另行特化)

特化必须在使用前可见,且不能在匿名命名空间里

最常踩的坑:把 std::formatter 特化放在 .cpp 文件里,或包在 namespace { } 中。链接期找不到特化,运行时报 std::format_error 或编译失败。

  • 特化必须与 std::formatter 在同一命名空间(即全局命名空间),且在首次调用 std::format 前完成声明和定义
  • 推荐放在头文件中,紧挨着 MyType 定义之后,或至少确保包含顺序正确
  • 如果 MyType 在命名空间 ns 中,特化仍必须写成 template struct std::formatter<:mytype char>,不能写 namespace ns { template struct std::formatter<...> }

复杂点在于 parse() 的健壮性和 format() 中对上下文输出器的精确控制——稍有不慎就会格式错乱或崩溃,而不是优雅降级。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

317

2023.08.02

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

618

2023.07.31

python中的format是什么意思
python中的format是什么意思

python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

430

2024.06.27

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

524

2023.09.20

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

257

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

208

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1465

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

619

2023.11.24

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Git 教程
Git 教程

共21课时 | 2.7万人学习

Git版本控制工具
Git版本控制工具

共8课时 | 1.5万人学习

Git中文开发手册
Git中文开发手册

共0课时 | 0人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号