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

直接用 std::hex 不会生成字符串,它只影响流输出格式;想得到十六进制字符串,得用 std::stringstream 或 C++17 的 std::to_chars,或者手动转换。
用 std::stringstream + std::hex 生成小写十六进制字符串
这是最常用、可读性好、兼容 C++11 的方式。注意:默认不补零,也不带 0x 前缀,需手动控制宽度和填充。
-
std::hex只对后续整数输出生效,不影响浮点数 - 要用
std::setw和std::setfill补零(例如固定 2 位:0a→0a,但5→05) - 小写是默认行为;大写需加
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 位,左补零 - 不支持负数的补码显示(会直接转为无符号等价值),如
-1在int上转成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 位十六进制 +\0) -
snprintf返回值是「欲写入长度」,不是「是否成功」,要检查是否截断 - 跨平台时,
long long的格式符不统一(%llxvs%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++ 标准、性能要求和是否需要精确控制格式——别只看“能转出来”,要看“转出来的字符串是否符合协议或调试预期”。











