c++中如何使用std::hex将整数转十六进制字符串_c++进制转换【汇总】

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

std::hex 本身不能直接转字符串,它只是流操纵器

很多人一看到 std::hex 就以为能像 Python 的 hex() 那样直接返回字符串,结果写 std::hex(123) 编译报错。实际上,std::hex 是个流格式化标志,必须配合 std::ostringstreamstd::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++11 起可用 sprintf(需确保缓冲区足够):
    char buf[16];
    sprintf(buf, "%02X", 15); // 得到 "0F"
    std::st

    ring s = "0x" + std::string(buf);
  • C++20 引入 std::format,语义清晰且类型安全:
    #include 
    std::string s = std::format("0x{:02X}", 15); // 直接得 "0X0F"
    (注意:GCC 13+ / Clang 15+ 默认支持,MSVC 2019 v16.10+)
  • 避免用 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 对象做多种进制输出,记得重置格式,否则会串扰。