std::bitset仅支持固定位宽二进制与整数互转:构造需指定模板参数,超长截断不报错;转十进制可能溢出抛异常;负数按补码解释;十六进制/八进制推荐用stringstream配合流操纵符。
std::bitset 做二进制 ↔ 十进制互转要注意位宽std::bitset 不是通用进制转换工具,它本质是固定长度的二进制位容器。想把一个整数转成二进制字符串,得先知道最大位宽:
std::bitset(123);若原数超长(如 123 对应 bitset 就溢出),高位直接截断,不报错也不警告.to_ulong() 或 .to_ullong(),但若位模式超出目标整型范围(如 33 位全 1 转 unsigned long),会抛 std::overflow_error
bitset(-1) 得到 11111111),不是符号位扩展逻辑std::stringstream + std::hex/std::oct 转十六进制、八进制这是最常用也最安全的整数 ↔ 其他进制字符串方案,依赖流操纵符控制输出格式:
ss 得到 "ff"(小写),加 std::uppercase 变 "FF"
ss 得到 "100"
0x 前缀;要加前缀得手动写 "0x" + ss.str()
stringstream 自动识别进制前缀;必须用 std::stoi(s, nullptr, base),其中 base=0 才支持 "0x" 和 "0" 前缀自动推断bitset,优先用 std::stoul(..., nullptr, 2)
如果输入是运行时得到的二进制字符串(如 "10101"),std::bitset 反而麻烦——你得先算长度再写模板参数,还不能处理可变长或带空格/前缀的输入:
std::stoul("10101", nullptr, 2),返回 21,简洁且支持任意长度(只要在 unsigned long 范围内)'0'/'1' 字符),stoul 会抛 std::invalid_argument;建议包在 try/catch 里std::stoul("1aF", nullptr, 16) → 431;八进制:std::stoul("755", nullptr, 8)
bitset 的
bitset(const std::string&) 构造函数虽能接受二进制字符串,但要求严格:只含 '0' 和 '1',且长度必须匹配模板参数,否则编译失败或运行时报 std::invalid_argument
实际项目中容易忽略三点:
stoul)在不同标准库实现中对超长字符串处理不一致;GCC libstdc++ 和 Clang libc++ 都支持远超 64 位的字符串,但 Windows MSVC 的 _stoul 可能早截断——关键逻辑别依赖“理论上支持”"00001010"),bitset 天然支持(bitset(10).to_string()),而 stoul → stringstream 流程需手动 std::setw(8)
stringstream 默认 locale 是 C,但如果全局 locale 被改过(如设成中文),std::hex 等操纵符可能失效;稳妥做法是显式绑定 ss.imbue(std::locale::classic())
std::stoul/std::stoull 配合 std::stringstream 是主力;std::bitset 只适合位宽确定、强调二进制语义(比如寄存器模拟、位操作掩码)的场景。别让它干解析字符串的活。