std::hex 不能直接转换整数为字符串,它只是需配合 std::ostringstream 等流对象使用的格式化操纵器;正确做法是构造字符串流、插入整数、设置 std::hex、再调用 .str() 获取结果。

std::hex 本身不能直接转字符串,它只是流操纵器
很多人一看到 std::hex 就以为能像 Python 的 hex() 那样直接返回字符串,结果写 std::hex(123) 编译报错。实际上,std::hex 是个流格式化标志,必须配合 std::ostringstream 或 std::cout 这类流对象使用,单独调用毫无意义。
正确路径是:构造一个字符串流 → 插入整数 → 设置 std::hex 格式 → 调用 .str() 获取字符串。
std::ostringstream oss; oss << std::hex << 255; std::string s = oss.str(); // 结果是 "ff"(小写,无前缀)
-
std::hex不影响大小写,输出默认小写;要大写得额外加std::uppercase - 不会自动补前导零,比如
15输出"f",不是"0f";需要补零得用std::setw+std::setfill - 不带
0x前缀;加前缀得手动拼接"0x" + s
想要大写、补零、带 0x 前缀?组合流操纵器更可靠
单靠 std::hex 功能太基础,实际项目中常需控制输出样式。C++ 流操纵器支持链式调用,把几个关键项串起来就能搞定常见需求:
std::ostringstream oss;
oss << std::hex << std::uppercase << std::setw(2) << std::setfill('0') << 15;
std::string s = "0x" + oss.str(); // 结果是 "0X0F"
-
std::uppercase:让 a–f 变成 A–F(注意:也把0x变成0X) -
std::setw(n):设置下一次输出的最小宽度(只对紧随其后的那个值生效) -
std::setfill(c):指定填充字符(通常和std::setw搭配用) - 如果要多次输出不同数字并都保持两位宽,每次都要重新写
std::setw(2),它不持久
性能敏感场景慎用 stringstream,考虑 sprintf 或 std::format(C++20)
频繁调用 std::ostringstream 构造/析构对象,在嵌入式或高频日志中可能成为瓶颈。这时可选更轻量的方式:
立即学习“C++免费学习笔记(深入)”;
- C++11 起可用
sprintf(需确保缓冲区足够):char buf[16]; sprintf(buf, "%02X", 15); // 得到 "0F" std::string s = "0x" + std::string(buf);
- C++20 引入
std::format,语义清晰且类型安全:#include
(注意:GCC 13+ / Clang 15+ 默认支持,MSVC 2019 v16.10+)std::string s = std::format("0x{:02X}", 15); // 直接得 "0X0F" - 避免用
itoa:它不是标准函数,Windows 专属,跨平台不可用
std::to_string 不支持进制转换,别误用
有人试图写 std::to_string(255, 16),这会编译失败——std::to_string 只接受一个参数,且只转十进制。它底层就是调用 std::sprintf 的十进制分支,和进制无关。
真正想“一行转十六进制字符串”,C++20 前最简方案仍是 std::ostringstream;C++20 起优先用 std::format,既安全又简洁。手写查表或位运算虽然最快,但易错、难维护,除非在裸机或极端性能场景,否则没必要。
最容易被忽略的是:流格式标志(如 std::hex)的作用域仅限当前流对象,且对后续插入操作持续生效,直到被显式覆盖(比如再插一个 std::dec)。如果复用同一个 std::ostringstream 对象做多种进制输出,记得重置格式,否则会串扰。











