c++中如何将十进制转换为十六进制字符串_c++ std::hex格式化【汇总】

11次阅读

用std::stringstream配合std::hex可生成小写十六进制字符串,兼容c++11;需手动用std::setw和std::setfill补零,不带0x前缀,默认小写,大写需std::uppercase。

c++中如何将十进制转换为十六进制字符串_c++ std::hex格式化【汇总】

直接用 std::hex 不会生成字符串,它只影响流输出格式;想得到十六进制字符串,得用 std::stringstream 或 C++17 的 std::to_chars,或者手动转换。

std::stringstream + std::hex 生成小写十六进制字符串

这是最常用、可读性好、兼容 C++11 的方式。注意:默认不补零,也不带 0x 前缀,需手动控制宽度和填充。

  • std::hex 只对后续整数输出生效,不影响浮点数
  • 要用 std::setwstd::setfill 补零(例如固定 2 位:0a0a,但 505
  • 小写是默认行为;大写需加 std::uppercase
std::stringstream ss; ss << std::hex << std::setw(2) << std::setfill('0') << 10; std::string hex_str = ss.str(); // "0a"

std::format(C++20)最简洁安全

如果你的编译器支持 C++20(如 GCC 13+、Clang 15+、MSVC 2022 17.3+),std::format 是首选:类型安全、无缓冲区风险、支持格式说明符。

  • {:x} → 小写,{:X} → 大写
  • {:02x} → 至少 2 位,左补零
  • 不支持负数的补码显示(会直接转为无符号等价值),如 -1int 上转成 ffffffff(32 位)
std::string s = std::format("{:04x}", 255); // "00ff" std::string t = std::format("{:X}", 255);     // "FF"

避免用 sprintf / snprintf 手动拼接

虽然能用,但容易出缓冲区溢出或格式错误,尤其处理不同整数宽度(int vs uint64_t)时,%x 对 64 位数在 windows 下可能失效(应为 %llx)。

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

  • 必须预先分配足够大的字符数组(比如 char buf[17] 才够存 64 位十六进制 +
  • snprintf 返回值是「欲写入长度」,不是「是否成功」,要检查是否截断
  • 跨平台时,long long 的格式符不统一(%llx vs %I64x
char buf[17]; int len = snprintf(buf, sizeof(buf), "%08x", 0x12345678); if (len < 0 || len >= (int)sizeof(buf)) { /* 错误或截断 */ }

std::to_chars(C++17)性能高但使用麻烦

它是唯一无内存分配、无异常、纯函数式的方式,适合高频/嵌入式场景,但需要手动管理缓冲区和结果检查。

  • 只支持无符号整型unsigned int, uint64_t 等),有符号数要先转为对应无符号类型
  • 返回 std::to_chars_result,需检查 ec 是否为 std::errc()
  • 不自动补零,也不支持大小写切换,全靠你控制起始位置和填充逻辑
char buf[16]; auto [ptr, ec] = std::to_chars(buf, buf + sizeof(buf), 255u, 16); if (ec == std::errc()) {     std::string s(buf, ptr); // "ff" }

真正要注意的是:不同方法对负数、高位零、大小写、类型宽度的处理差异极大,选哪种取决于你的 C++ 标准、性能要求和是否需要精确控制格式——别只看“能转出来”,要看“转出来的字符串是否符合协议或调试预期”。

text=ZqhQzanResources