当前位置: 首页 > news >正文

C++中的格式化字符串

C++中的格式化字符串

fmt 库简介

{fmt}是一个开源的、现代化的C++格式化库,由Victor Zverovich创建。它提供了一种安全、快速且方便的字符串格式化方式,其设计理念受到了Python的str.format()的启发

fmt库的主要特点

  1. 易用性:使用简洁的语法,基于花括号{}的占位符
  2. 类型安全:在编译时检查格式字符串和参数类型匹配
  3. 高性能:比标准库的printfiostream更快
  4. 可扩展:支持自定义类型的格式化
  5. 丰富的格式化选项:对齐、填充、精度等控制

Ubuntu下安装fmt库

使用下面的命令进行安装:

sudo apt install libfmt-dev

安装完成后,可以检查fmt库是否正确安装。例如,查看头文件是否存在:

ls /usr/include/fmt

如果目录存在且包含相关头文件,则说明安装成功

也可以通过下面的命令查看fmt库版本:

fmt --version

基本格式化

fmt库提供了简单直观的基本格式化功能,使用花括号{}作为参数占位符:

#include <fmt/core.h>
#include <string>int main() 
{int number = 42;double pi = 3.14159;std::string name = "C++";// 基本格式化std::string result = fmt::format("数字是 {}", number);// 结果: "数字是 42"// 多个参数按顺序替换,类似于printfstd::string result2 = fmt::format("{} 的值约为 {}", name, pi);// 结果: "C++ 的值约为 3.14159"// 直接输出fmt::print("程序运行结果: {}\n", "成功");// 输出: "程序运行结果: 成功"
}

基本格式化遵循参数与占位符按位置一一对应的原则,简单易用且类型安全

带索引的参数

fmt库支持通过索引指定参数顺序,索引从0开始,可以重复使用同一参数:

#include <fmt/core.h>
#include <string>int main() 
{int x = 10;int y = 20;std::string language = "C++";// 使用索引指定参数std::string result = fmt::format("{0} + {1} = {2}", x, y, x + y);// 结果: "10 + 20 = 30"// 重复使用同一参数std::string result2 = fmt::format("{0}真是太强大了!我爱{0}!", language);// 结果: "C++真是太强大了!我爱C++!"// 改变参数顺序// 即0代表第一个参数,1代表第二个参数// 用下标位置代表替换位置std::string result3 = fmt::format("坐标:({1}, {0})", x, y);// 结果: "坐标:(20, 10)"// 混合使用索引和自动位置// 注意: 一旦使用了索引,所有占位符都应使用索引std::string result4 = fmt::format("{0}的值是{1}", "π", 3.14159);// 结果: "π的值是3.14159"
}

带索引的参数格式化使代码更灵活,特别是在需要重复使用某个参数或调整参数顺序时非常有用

格式说明符

基本介绍

格式说明符让您可以控制数据的呈现方式,语法为{[参数位置]:[格式说明]}

#include <fmt/core.h>
#include <string>int main() 
{int number = 42;double pi = 3.14159265359;// 数值进制表示std::string hex = fmt::format("十六进制: {:#x}", number);  // "十六进制: 0x2a"std::string oct = fmt::format("八进制: {:#o}", number);    // "八进制: 052"std::string bin = fmt::format("二进制: {:#b}", number);    // "二进制: 0b101010"// 浮点数精度std::string precise = fmt::format("π值保留两位小数: {:.2f}", pi);  // "π值保留两位小数: 3.14"std::string scientific = fmt::format("科学计数法: {:.2e}", pi);     // "科学计数法: 3.14e+00"// 宽度和对齐std::string right = fmt::format("右对齐:|{:10}|", "文本");     // "右对齐:|      文本|"std::string left = fmt::format("左对齐:|{:<10}|", "文本");     // "左对齐:|文本      |"std::string center = fmt::format("居中:|{:^10}|", "文本");     // "居中:|   文本   |"// 填充字符// 中文占两个字符,剩余11个字符用*填充,总共15个字符std::string fill = fmt::format("|{:*^15}|", "标题");          // "|*****标题******|"// 符号控制std::string always_sign = fmt::format("{:+d}", 42);           // "+42"std::string space_sign = fmt::format("{: d}", 42);            // " 42"// 千位分隔符std::string grouped = fmt::format("{:L}", 1000000);           // "1,000,000" (依赖地区设置)// 多种格式组合std::string combined = fmt::format("{:*^+10.2f}", 42.1234);   // "**+42.12**"return 0;
}
常用格式说明符

格式说明符的完整语法是:{[参数索引]:[填充字符][对齐][符号][#][0][宽度][.精度][类型]}

  • 填充和对齐

    • <:左对齐
    • >:右对齐(默认)
    • ^:居中
  • 符号

    • +:总是显示正负号
    • -:只对负数显示负号(默认)
    • 空格:正数前加空格,负数前加负号
  • 类型

    • d:十进制整数
    • x/X:小写/大写十六进制
    • o:八进制
    • b/B:小写/大写二进制
    • f/F:固定小数点
    • e/E:科学计数法
    • g/G:通用格式(自动选择 f 或 e)
    • s:字符串
  • 特殊标记

    • #:显示前缀(0x、0b 等)或小数点
    • 0:用零填充数字

fmt库的格式说明符系统非常强大且灵活,可以满足各种复杂的字符串格式化需求

fmt的高级特性

  1. 自定义类型格式化

    struct Point {int x, y;
    };template <> 
    struct fmt::formatter<Point> {constexpr auto parse(format_parse_context& ctx) { return ctx.begin(); }template <typename FormatContext>auto format(const Point& p, FormatContext& ctx) {return fmt::format_to(ctx.out(), "({}, {})", p.x, p.y);}
    };// 使用
    Point p{10, 20};
    fmt::print("点坐标: {}", p);  // 输出: 点坐标: (10, 20)
    
  2. 颜色支持

    #include <fmt/color.h>fmt::print(fg(fmt::color::red), "这段文字是红色的\n");
    fmt::print(bg(fmt::color::blue) | fg(fmt::color::white), "蓝底白字\n");
    
  3. 格式化到容器或内存

    std::vector<char> out;
    fmt::format_to(std::back_inserter(out), "Hello, {}!", "world");char buffer[100];
    auto result = fmt::format_to_n(buffer, 100, "The answer is {}.", 42);
    

std::format(C++20)

std::format是C++20引入的标准库功能,其设计和API直接基于{fmt}

基本用法

#include <format>
#include <string>int main()
{int answer = 42;std::string name = "C++";// 基本格式化std::string s = std::format("Hello, {}! The answer is {}", name, answer);// 带格式说明符std::string s2 = std::format("十进制: {0:d}, 十六进制: {0:#x}, 八进制: {0:#o}", answer);// 对齐和填充// Unicode编码,中文占3个字符std::string s3 = std::format("{:*^10}", "居中"); // **居中**return 0;
}

常用格式说明符

格式说明符的语法为:{[参数位置]:[填充][对齐][符号][#][0][宽度][.精度][类型]}

  • 填充和对齐< (左对齐),> (右对齐),^ (居中)

    std::format("{:<10}", "左对齐");   // "左对齐 "
    std::format("{:>10}", "右对齐");   // " 右对齐"
    std::format("{:^10}", "居中");     // "  居中  "
    std::format("{:*^10}", "居中");    // "**居中**"
    
  • 符号+ (总是显示符号),- (仅负数显示符号), (空格,正数前加空格)

    std::format("{:+}", 42);    // "+42"
    std::format("{: }", 42);    // " 42"
    
  • 数值格式

    std::format("{:d}", 42);    // 十进制: "42"
    std::format("{:#x}", 42);   // 十六进制带前缀: "0x2a"
    std::format("{:.2f}", 3.14159); // 浮点数保留2位小数: "3.14"
    

fmt与std::format的对比

相似点

  1. 语法:两者使用相同的基于{}的格式语法
  2. 类型安全:都在编译时进行类型检查
  3. API设计:std::format直接基于fmt库设计

不同点

  1. 可用性:fmt可用于任何C++版本,而std::format需要C++20
  2. 功能集:fmt提供更多高级功能(颜色、I/O 等)
  3. 性能:由于不同的实现,性能可能有细微差异
  4. 命名空间:fmt使用fmt::命名空间,标准库使用std::

代码兼容性示例

可以编写同时支持两种库的代码:

#if __cplusplus >= 202002L && defined(__cpp_lib_format)#include <format>namespace fmt_ns = std;
#else#include <fmt/core.h>namespace fmt_ns = fmt;
#endifstd::string message = fmt_ns::format("值: {}", 42);

实际应用场景

  1. 日志系统:格式化各类日志消息
  2. 用户界面:生成格式化的显示文本
  3. 配置文件生成:创建具有特定格式的配置
  4. 报告生成:生成格式化的数据报告
  5. 错误信息:创建详细的错误描述
http://www.xdnf.cn/news/197155.html

相关文章:

  • TMI投稿指南(二):投稿文章注意事项
  • JAVA设计模式——(八)单例模式
  • 详解UnityWebRequest类
  • 02_使用 AES 算法实现文件加密上传至阿里云、解密下载
  • 阿里云ftp服务器登录要怎么做?如何访问ftp服务器?
  • 【权限模型】RBAC模型详解
  • 东莞SMT贴片加工工艺优化解析
  • 青年座谈会的读书汇报
  • Kotlin DSL 深度解析:从 Groovy 迁移的困惑与突破
  • vue3 如何设置全局属性
  • Docker的常用命令
  • WPF之Button控件详解
  • Crusader Kings III 王国风云 3(十字军之王 3) [DLC 解锁] [Steam] [Windows SteamOS macOS]
  • 多元函数微分之传统方法和全微分法
  • Qt笔记-自定义托盘弹出界面
  • 用于表格数据的神经网络分类器_TabPFN
  • Vuex持续保存数据 页面刷新不丢失 vuex-persistedstate
  • WPF 程序监控硬件设备状态变化的实现方案
  • vscode源代码管理Tab-文件右侧标志(M、A 等)的含义
  • await和async
  • (28)VTK C++开发示例 ---添加坐标轴
  • std::print 和 std::println
  • echarts自定义图表--仪表盘
  • 【机器学习】人工智能在电力电子领域的应用
  • 反射与注解实现动态功能扩展案例-插件系统
  • 模型上下文协议(MCP)深度解析:大模型从“思考者“进化为“行动者“
  • 总账主数据——Part 2 科目-4
  • 优先队列和单调队列(双端队列实现的)
  • stm32wb55rg (1) 基于IAR 创建stm32 工程模板
  • 序列密码算法ShanLooog512设计原理详解