**{fmt}**是一个开源格式化库,为C stdio和C++ iostreams提供了快速且安全的替代方案。
如果您喜欢这个项目,请考虑向帮助乌克兰战争受害者的基金捐款:https://www.stopputin.net/。
在Compiler Explorer中试用{fmt}。
(s)printf、iostreams、to_string和to_chars更快,参见速度测试和每秒将一亿个整数转换为字符串core.h、format.h和format-inl.h;参见编译时间和代码膨胀-Wall -Wextra -pedantic)下也无警告的干净代码库FMT_HEADER_ONLY宏启用更多详情请参阅文档。
打印到标准输出 (运行)
#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的现代终端上的输出:
| 库 | 方法 | 运行时间,秒 |
|---|---|---|
| libc | printf | 0.91 |
| libc++ | std::ostream | 2.49 |
| {fmt} 9.1 | fmt::print | 0.74 |
| Boost Format 1.80 | boost::format | 6.26 |
| Folly Format | folly::format | 1.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 float 和 double 格式化方面,{fmt} 比 std::ostringstream 和 sprintf 快 20-30 倍(dtoa-benchmark),并且比 double-conversion 和 ryu 更快:
[图片]
format-benchmark 中的脚本 bloat-test.py 测试了非平凡项目的编译时间和代码膨胀。它生成 100 个翻译单元,每个单元中使用 printf() 或其替代品五次,以模拟中等规模的项目。结果可执行文件大小和编译时间(使用 Apple clang 版本 15.0.0 (clang-1500.1.0.2.5),macOS Sonoma,取三次中的最佳结果)如下表所示。
优化构建 (-O3)
| 方法 | 编译时间,秒 | 可执行文件大小,KiB | 剥离后大小,KiB |
|---|---|---|---|
| printf | 1.6 | 54 | 50 |
| IOStreams | 25.9 | 98 | 84 |
| fmt 83652df | 4.8 | 54 | 50 |
| tinyformat | 29.1 | 161 | 136 |
| Boost Format | 55.0 | 530 | 317 |
{fmt} 编译速度快,在每次调用的二进制大小方面与 printf 相当(在此系统上误差在四舍五入范围内)。
非优化构建
| 方法 | 编译时间,秒 | 可执行文件大小,KiB | 剥离后大小,KiB |
|---|---|---|---|
| printf | 1.4 | 54 | 50 |
| IOStreams | 23.4 | 92 | 68 |
| {fmt} 83652df | 4.4 | 89 | 85 |
| tinyformat | 24.5 | 204 | 161 |
| Boost Format | 36.4 | 831 | 462 |
libc、lib(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 检查,如果配置为这样做,它能够将 printf 和 fprintf 的出现转换为 fmt::print。(默认情况下,它转换为 std::print。)
如果您知道其他使用本库的项目,请通过电子邮件或提交issue告诉我。
那么为什么还要开发另一个格式化库呢?
有很多方法可以完成这个任务,从标准方法如printf系列函数和iostreams到Boost Format和FastFormat库。创建新库的原因是我发现的每个现有解决方案要么存在严重问题,要么无法提供我需要的所有功能。
printf的优点是速度快,作为C标准库的一部分,随时可用。主要缺点是不支持用户定义类型。printf还有安全问题,尽管在GCC中通过attribute ((format (printf, ...)))在某种程度上得到了缓解。有一个POSIX扩展为printf添加了i18n所需的位置参数,但它不是C99的一部分,可能在某些平台上不可用。
iostreams的主要问题最好用一个例子来说明:
std::cout << std::setprecision(2) << std::fixed << 1.23456 << "\n";
与printf相比,这需要大量输入:
printf("%.2f\n", 1.23456);
FastFormat的作者Matthew Wilson称之为"尖括号地狱"。iostreams在设计上不支持位置参数。
好处是iostreams支持用户定义类型,并且安全,尽管错误处理很笨拙。
这是一个非常强大的库,支持类似printf的格式字符串和位置参数。它的主要缺点是性能。根据各种基准测试,它比这里考虑的其他方法慢得多。Boost Format还有过长的构建时间和严重的代码膨胀问题(见基准测试)。
这是一个有趣的库,速度快、安全,并且有位置参数。然而,它有重大限制,引用其作者的话:
三个在当前设计中无法容纳的功能是:
- 前导零(或任何其他非空格填充)
- 八进制/十六进制编码
- 运行时宽度/对齐规范
它也相当大,并且严重依赖STLSoft,这可能对某些项目的使用过于限制。
这不是一个格式化库,但我决定在这里包含它以求完整。与iostreams一样,它存在将文字文本与参数混合的问题。该库相当快,但在Karma自己的基准测试中,整数格式化速度比使用格式字符串编译的fmt::format_to慢,参见每秒将一亿个整数转换为字符串。
{fmt}在MIT许可证下分发。
文档中的格式字符串语法部分基于Python 字符串模块文档中的内容。因此,文档在doc/python-license.txt中提供的Python软件基金会许可证下分发。它仅适用于分发{fmt}的文档。
{fmt}库由Victor Zverovich(vitaut)维护,并得到了许多其他人的贡献。部分贡献者和版本信息可以在贡献者和发布页面查看。如果您的贡献未被列出或提及有误,请告知我们,我们会进行修正。
如需报告安全问题,请在安全公告页面披露。
本项目由志愿者团队在合理努力的基础上维护。因此,在公开披露之前,请给我们至少90天的时间来修复问题。


稳定高效的流量提升解决方案,助力品牌曝光
稳定高效的流量提升解决方案,助力品牌曝光


最新版Sora2模型免费使用,一键生成无水印视频
最新版Sora2模型免费使用,一键生成无水印视频


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


选题、配图、成文,一站式创作,让内容运营更高效
讯飞绘文,一个AI集成平台,支持写作、选题、配图、排版和发布。高效生成适用于各类媒体的定制内容,加速品牌传播,提升内容营销效果。


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


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


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


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


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


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

微信扫一扫关注公众号