必须设置 std::hex 和 std::showbase 才能将 int 转为带 0x 前缀的十六进制字符串,否则默认输出为十进制且无前缀。

用 std::stringstream 转 int 为十六进制字符串,必须设 std::hex 和 std::showbase
默认情况下,std::stringstream 输出整数是十进制,不加修饰直接 int 不会变成 "0x1a" 这样的字符串。要得到十六进制表示,得显式设置格式标志:
-
std::hex:启用十六进制输出(小写字母 a–f) -
std::showbase:添加前缀0x(不加这个,输出只是1a,不是0x1a) - 如果需要大写(
0X1A),再加std::uppercase
示例:
#include#include #include int x = 26; std::stringstream ss; ss << std::hex << std::showbase << x; std::string s = ss.str(); // s == "0x1a"
std::to_string 不能直接转十六进制,别误用
std::to_string 只支持十进制转换,传入 int 永远返回十进制字符串。它不接受任何进制参数,也没有重载支持 std::hex。下面这段代码是错的:
std::string s = std::to_string(std::hex, 26); // 编译失败!
常见误判点:
立即学习“C++免费学习笔记(深入)”;
- 以为
std::to_string和 Python 的hex()类似 —— 实际完全无关 - 试图在
to_string后链式调用—— 语法不成立,to_string返回的是std::string,不是流对象
用 std::format(C++20)更简洁,但注意编译器支持
C++20 引入了 std::format,一行就能搞定,且可选大小写和宽度:
#includestd::string s1 = std::format("{:#x}", 26); // "0x1a" std::string s2 = std::format("{:#X}", 26); // "0X1A" std::string s3 = std::format("{:04x}", 26); // "001a"(补零到4位)
但要注意:
- MSVC 从 19.30+、GCC 13+、Clang 15+ 才完整支持
std::format - 某些旧项目或嵌入式环境仍禁用 C++20,此时只能退回
stringstream或sprintf -
{:#x}中的#对应showbase,不可省略,否则输出是1a而非0x1a
避免漏掉 std::setw 和 std::setfill 导致补零失效
想把 26 格式化成固定宽度如 "0000001a",只设 std::hex 是不够的。必须配合 std::setw(仅对下一次输出生效)和 std::setfill:
ss.str(""); // 清空
ss << std::hex << std::showbase << std::setw(8) << std::setfill('0') << 26;
// 结果是 "0x00001a" —— 注意:showbase 会占用前两位,所以 setw(8) 实际填充的是后面6位
容易踩的坑:
-
std::setw不是持久的,每次输出前都得重新设 - 如果先写
showbase再写setw,填充位置在0x后面,不是整个字符串左端 - 要真正实现 "0000001a"(无
0x),就得去掉showbase,再用std::setw+std::setfill
std::format 最省心,但兼容性差;stringstream 最通用,但容易忘设标志或搞混填充顺序。最常被忽略的是:不加 showbase 就得不到 0x 前缀,而很多人默认以为 hex 就该带前缀。











