fmt

fmt

高效安全的C++文本格式化库

fmt是一个开源C++格式化库,为C stdio和C++ iostreams提供高效安全的替代方案。它具有简洁的API、C++20 std::format实现、Unicode支持和安全的printf功能。fmt不仅性能卓越、代码精简,还易于使用且可靠性高,是C++项目进行文本格式化的理想工具。

fmtC++格式化库性能跨平台Github开源项目
<img src="https://yellow-cdn.veclightyear.com/ab5030c0/e7426cef-3a69-44b7-a484-01d8fabca8ad.png" alt="{fmt}" width="25%"/>

image image image fmt在oss-fuzz持续进行模糊测试 在StackOverflow上使用fmt标签提问 image

**{fmt}**是一个开源格式化库,为C stdio和C++ iostreams提供了快速且安全的替代方案。

如果您喜欢这个项目,请考虑向帮助乌克兰战争受害者的基金捐款:https://www.stopputin.net/

文档

速查表

问答:在StackOverflow上使用fmt标签提问

Compiler Explorer中试用{fmt}。

特性

更多详情请参阅文档

示例

打印到标准输出 (运行)

#include <fmt/core.h> int main() { fmt::print("Hello, world!\n"); }

格式化字符串 (运行)

std::string s = fmt::format("The answer is {}.", 42); // s == "The answer is 42."

使用位置参数格式化字符串 (运行)

std::string s = fmt::format("I'd rather be {1} than {0}.", "right", "happy"); // s == "I'd rather be happy than right."

打印日期和时间 (运行)

#include <fmt/chrono.h> int main() { auto now = std::chrono::system_clock::now(); fmt::print("Date and time: {}\n", now); fmt::print("Time: {:%H:%M}\n", now); }

输出:

Date and time: 2023-12-26 19:10:31.557195597
Time: 19:10

打印容器 (运行)

#include <vector> #include <fmt/ranges.h> int main() { std::vector<int> v = {1, 2, 3}; fmt::print("{}\n", v); }

输出:

[1, 2, 3]

在编译时检查格式字符串

std::string s = fmt::format("{:d}", "I am not a number");

在C++20中,这会产生编译时错误,因为d对字符串来说是无效的格式说明符。

从单线程写入文件

#include <fmt/os.h> int main() { auto out = fmt::output_file("guide.txt"); out.print("Don't {}", "Panic"); }

这可能比fprintf快5到9倍

使用颜色和文本样式打印

#include <fmt/color.h> int main() { fmt::print(fg(fmt::color::crimson) | fmt::emphasis::bold, "Hello, {}!\n", "world"); fmt::print(fg(fmt::color::floral_white) | bg(fmt::color::slate_gray) | fmt::emphasis::underline, "Olá, {}!\n", "Mundo"); fmt::print(fg(fmt::color::steel_blue) | fmt::emphasis::italic, "你好{}!\n", "世界"); }

在支持Unicode的现代终端上的输出:

image

基准测试

速度测试

方法运行时间,秒
libcprintf0.91
libc++std::ostream2.49
{fmt} 9.1fmt::print0.74
Boost Format 1.80boost::format6.26
Folly Formatfolly::format1.87

在基准测试的方法中,{fmt}是最快的,比printf快约20%。 上述结果是通过在 macOS 12.6.1 上使用 clang++ -O3 -DNDEBUG -DSPEED_TEST -DHAVE_FORMAT 构建 tinyformat_test.cpp 并取三次运行中的最佳结果得出的。在测试中,格式字符串 "%0.10f:%04d:%+g:%s:%p:%c:%%\n" 或等效格式被填充 2,000,000 次,输出发送到 /dev/null;更多详情请参考源代码

在 IEEE754 floatdouble 格式化方面,{fmt} 比 std::ostringstreamsprintf 快 20-30 倍(dtoa-benchmark),并且比 double-conversionryu 更快:

[图片]

编译时间和代码膨胀

format-benchmark 中的脚本 bloat-test.py 测试了非平凡项目的编译时间和代码膨胀。它生成 100 个翻译单元,每个单元中使用 printf() 或其替代品五次,以模拟中等规模的项目。结果可执行文件大小和编译时间(使用 Apple clang 版本 15.0.0 (clang-1500.1.0.2.5),macOS Sonoma,取三次中的最佳结果)如下表所示。

优化构建 (-O3)

方法编译时间,秒可执行文件大小,KiB剥离后大小,KiB
printf1.65450
IOStreams25.99884
fmt 83652df4.85450
tinyformat29.1161136
Boost Format55.0530317

{fmt} 编译速度快,在每次调用的二进制大小方面与 printf 相当(在此系统上误差在四舍五入范围内)。

非优化构建

方法编译时间,秒可执行文件大小,KiB剥离后大小,KiB
printf1.45450
IOStreams23.49268
{fmt} 83652df4.48985
tinyformat24.5204161
Boost Format36.4831462

libclib(std)c++libfmt 都作为共享库链接,以仅比较格式化函数的开销。Boost Format 是一个仅头文件的库,因此不提供任何链接选项。

运行测试

关于如何构建库和运行单元测试的说明,请参阅从源代码构建

基准测试位于单独的仓库 format-benchmarks 中,因此要运行基准测试,您首先需要克隆此仓库并使用 CMake 生成 Makefiles:

$ git clone --recursive https://github.com/fmtlib/format-benchmark.git
$ cd format-benchmark
$ cmake .

然后您可以运行速度测试:

$ make speed-test

或膨胀测试:

$ make bloat-test

代码迁移

clang-tidy v18 提供了 modernize-use-std-print 检查,如果配置为这样做,它能够将 printffprintf 的出现转换为 fmt::print。(默认情况下,它转换为 std::print。)

使用此库的著名项目

  • 0 A.D.:一款免费、开源、跨平台的即时战略游戏
  • AMPL/MP:一个用于数学规划的开源库
  • Apple的FoundationDB:一个开源的分布式事务型键值存储
  • Aseprite:动画精灵编辑器和像素艺术工具
  • AvioBook:一套全面的飞机操作套件
  • 暴雪战网:一个在线游戏平台
  • Celestia:太空实时3D可视化
  • Ceph:一个可扩展的分布式存储系统
  • ccache:一个编译器缓存
  • ClickHouse:一个分析型数据库管理系统
  • Contour:一个现代终端模拟器
  • CUAUV:康奈尔大学的自主水下航行器
  • Drake:非线性动力系统的规划、控制和分析工具箱(麻省理工学院)
  • Envoy:C++编写的L7代理和通信总线(Lyft)
  • FiveM:GTA V的修改框架
  • fmtlog:一个高性能的fmtlib风格日志库,延迟在纳秒级
  • Folly:Facebook开源库
  • GemRB:Bioware的无限引擎的可移植开源实现
  • Grand Mountain Adventure:一款美丽的开放世界滑雪和单板滑雪游戏
  • HarpyWar/pvpgn:玩家对战游戏网络,带有优化
  • KBEngine:一个开源的MMOG服务器引擎
  • Keypirinha:Windows的语义启动器
  • Kodi(前身为xbmc):家庭影院软件
  • Knuth:高性能比特币全节点
  • libunicode:一个现代C++17 Unicode库
  • MariaDB:关系数据库管理系统
  • Microsoft Verona:并发所有权研究编程语言
  • MongoDB:分布式文档数据库
  • MongoDB Smasher:生成随机数据集的小工具
  • OpenSpace:一个开源的天体可视化框架
  • PenUltima Online (POL):一个MMO服务器,兼容大多数Ultima Online客户端
  • PyTorch:一个开源机器学习库
  • quasardb:一个分布式、高性能的关联数据库
  • Quill:异步低延迟日志库
  • QKW:通过别名简化导航,执行复杂的多行终端命令序列
  • redis-cerberus:Redis集群代理
  • redpanda:用C++编写的比Kafka®快10倍的关键任务系统替代品
  • rpclib:现代C++ msgpack-RPC服务器和客户端库
  • Salesforce Analytics Cloud:商业智能软件
  • Scylla:兼容Cassandra的NoSQL数据存储,单服务器可处理每秒100万事务
  • Seastar:用于现代硬件上高性能服务器应用的先进开源C++框架
  • spdlog:超快的C++日志库
  • Stellar:金融平台
  • Touch Surgery:手术模拟器
  • TrinityCore:开源MMORPG框架
  • 🐙 userver框架:具有丰富抽象和数据库驱动程序的开源异步框架
  • Windows Terminal:新的Windows终端

更多...

如果您知道其他使用本库的项目,请通过电子邮件或提交issue告诉我。

动机

那么为什么还要开发另一个格式化库呢?

有很多方法可以完成这个任务,从标准方法如printf系列函数和iostreams到Boost Format和FastFormat库。创建新库的原因是我发现的每个现有解决方案要么存在严重问题,要么无法提供我需要的所有功能。

printf

printf的优点是速度快,作为C标准库的一部分,随时可用。主要缺点是不支持用户定义类型。printf还有安全问题,尽管在GCC中通过attribute ((format (printf, ...)))在某种程度上得到了缓解。有一个POSIX扩展为printf添加了i18n所需的位置参数,但它不是C99的一部分,可能在某些平台上不可用。

iostreams

iostreams的主要问题最好用一个例子来说明:

std::cout << std::setprecision(2) << std::fixed << 1.23456 << "\n";

与printf相比,这需要大量输入:

printf("%.2f\n", 1.23456);

FastFormat的作者Matthew Wilson称之为"尖括号地狱"。iostreams在设计上不支持位置参数。

好处是iostreams支持用户定义类型,并且安全,尽管错误处理很笨拙。

Boost Format

这是一个非常强大的库,支持类似printf的格式字符串和位置参数。它的主要缺点是性能。根据各种基准测试,它比这里考虑的其他方法慢得多。Boost Format还有过长的构建时间和严重的代码膨胀问题(见基准测试)。

FastFormat

这是一个有趣的库,速度快、安全,并且有位置参数。然而,它有重大限制,引用其作者的话:

三个在当前设计中无法容纳的功能是:

  • 前导零(或任何其他非空格填充)
  • 八进制/十六进制编码
  • 运行时宽度/对齐规范

它也相当大,并且严重依赖STLSoft,这可能对某些项目的使用过于限制。

Boost Spirit.Karma

这不是一个格式化库,但我决定在这里包含它以求完整。与iostreams一样,它存在将文字文本与参数混合的问题。该库相当快,但在Karma自己的基准测试中,整数格式化速度比使用格式字符串编译的fmt::format_to慢,参见每秒将一亿个整数转换为字符串

许可证

{fmt}在MIT许可证下分发。

文档许可证

文档中的格式字符串语法部分基于Python 字符串模块文档中的内容。因此,文档在doc/python-license.txt中提供的Python软件基金会许可证下分发。它仅适用于分发{fmt}的文档。

维护者

{fmt}库由Victor Zverovich(vitaut)维护,并得到了许多其他人的贡献。部分贡献者和版本信息可以在贡献者发布页面查看。如果您的贡献未被列出或提及有误,请告知我们,我们会进行修正。

安全政策

如需报告安全问题,请在安全公告页面披露。

本项目由志愿者团队在合理努力的基础上维护。因此,在公开披露之前,请给我们至少90天的时间来修复问题。

编辑推荐精选

商汤小浣熊

商汤小浣熊

最强AI数据分析助手

小浣熊家族Raccoon,您的AI智能助手,致力于通过先进的人工智能技术,为用户提供高效、便捷的智能服务。无论是日常咨询还是专业问题解答,小浣熊都能以快速、准确的响应满足您的需求,让您的生活更加智能便捷。

imini AI

imini AI

像人一样思考的AI智能体

imini 是一款超级AI智能体,能根据人类指令,自主思考、自主完成、并且交付结果的AI智能体。

Keevx

Keevx

AI数字人视频创作平台

Keevx 一款开箱即用的AI数字人视频创作平台,广泛适用于电商广告、企业培训与社媒宣传,让全球企业与个人创作者无需拍摄剪辑,就能快速生成多语言、高质量的专业视频。

即梦AI

即梦AI

一站式AI创作平台

提供 AI 驱动的图片、视频生成及数字人等功能,助力创意创作

扣子-AI办公

扣子-AI办公

AI办公助手,复杂任务高效处理

AI办公助手,复杂任务高效处理。办公效率低?扣子空间AI助手支持播客生成、PPT制作、网页开发及报告写作,覆盖科研、商业、舆情等领域的专家Agent 7x24小时响应,生活工作无缝切换,提升50%效率!

TRAE编程

TRAE编程

AI辅助编程,代码自动修复

Trae是一种自适应的集成开发环境(IDE),通过自动化和多元协作改变开发流程。利用Trae,团队能够更快速、精确地编写和部署代码,从而提高编程效率和项目交付速度。Trae具备上下文感知和代码自动完成功能,是提升开发效率的理想工具。

热门AI工具生产力协作转型TraeAI IDE
蛙蛙写作

蛙蛙写作

AI小说写作助手,一站式润色、改写、扩写

蛙蛙写作—国内先进的AI写作平台,涵盖小说、学术、社交媒体等多场景。提供续写、改写、润色等功能,助力创作者高效优化写作流程。界面简洁,功能全面,适合各类写作者提升内容品质和工作效率。

AI助手AI工具AI写作工具AI辅助写作蛙蛙写作学术助手办公助手营销助手
问小白

问小白

全能AI智能助手,随时解答生活与工作的多样问题

问小白,由元石科技研发的AI智能助手,快速准确地解答各种生活和工作问题,包括但不限于搜索、规划和社交互动,帮助用户在日常生活中提高效率,轻松管理个人事务。

聊天机器人AI助手热门AI工具AI对话
Transly

Transly

实时语音翻译/同声传译工具

Transly是一个多场景的AI大语言模型驱动的同声传译、专业翻译助手,它拥有超精准的音频识别翻译能力,几乎零延迟的使用体验和支持多国语言可以让你带它走遍全球,无论你是留学生、商务人士、韩剧美剧爱好者,还是出国游玩、多国会议、跨国追星等等,都可以满足你所有需要同传的场景需求,线上线下通用,扫除语言障碍,让全世界的语言交流不再有国界。

讯飞智文

讯飞智文

一键生成PPT和Word,让学习生活更轻松

讯飞智文是一个利用 AI 技术的项目,能够帮助用户生成 PPT 以及各类文档。无论是商业领域的市场分析报告、年度目标制定,还是学生群体的职业生涯规划、实习避坑指南,亦或是活动策划、旅游攻略等内容,它都能提供支持,帮助用户精准表达,轻松呈现各种信息。

热门AI工具AI办公办公工具讯飞智文AI在线生成PPTAI撰写助手多语种文档生成AI自动配图
下拉加载更多