Project Icon

fmt

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

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

{fmt}

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天的时间来修复问题。

项目侧边栏1项目侧边栏2
推荐项目
Project Cover

豆包MarsCode

豆包 MarsCode 是一款革命性的编程助手,通过AI技术提供代码补全、单测生成、代码解释和智能问答等功能,支持100+编程语言,与主流编辑器无缝集成,显著提升开发效率和代码质量。

Project Cover

AI写歌

Suno AI是一个革命性的AI音乐创作平台,能在短短30秒内帮助用户创作出一首完整的歌曲。无论是寻找创作灵感还是需要快速制作音乐,Suno AI都是音乐爱好者和专业人士的理想选择。

Project Cover

白日梦AI

白日梦AI提供专注于AI视频生成的多样化功能,包括文生视频、动态画面和形象生成等,帮助用户快速上手,创造专业级内容。

Project Cover

有言AI

有言平台提供一站式AIGC视频创作解决方案,通过智能技术简化视频制作流程。无论是企业宣传还是个人分享,有言都能帮助用户快速、轻松地制作出专业级别的视频内容。

Project Cover

Kimi

Kimi AI助手提供多语言对话支持,能够阅读和理解用户上传的文件内容,解析网页信息,并结合搜索结果为用户提供详尽的答案。无论是日常咨询还是专业问题,Kimi都能以友好、专业的方式提供帮助。

Project Cover

讯飞绘镜

讯飞绘镜是一个支持从创意到完整视频创作的智能平台,用户可以快速生成视频素材并创作独特的音乐视频和故事。平台提供多样化的主题和精选作品,帮助用户探索创意灵感。

Project Cover

讯飞文书

讯飞文书依托讯飞星火大模型,为文书写作者提供从素材筹备到稿件撰写及审稿的全程支持。通过录音智记和以稿写稿等功能,满足事务性工作的高频需求,帮助撰稿人节省精力,提高效率,优化工作与生活。

Project Cover

阿里绘蛙

绘蛙是阿里巴巴集团推出的革命性AI电商营销平台。利用尖端人工智能技术,为商家提供一键生成商品图和营销文案的服务,显著提升内容创作效率和营销效果。适用于淘宝、天猫等电商平台,让商品第一时间被种草。

Project Cover

AIWritePaper论文写作

AIWritePaper论文写作是一站式AI论文写作辅助工具,简化了选题、文献检索至论文撰写的整个过程。通过简单设定,平台可快速生成高质量论文大纲和全文,配合图表、参考文献等一应俱全,同时提供开题报告和答辩PPT等增值服务,保障数据安全,有效提升写作效率和论文质量。

投诉举报邮箱: service@vectorlightyear.com
@2024 懂AI·鲁ICP备2024100362号-6·鲁公网安备37021002001498号