按位取反字符串应逐字节异或0xFF:先转unsigned char避免符号扩展,再异或0xFF,最后转回char;禁用~操作符以防截断和不可移植问题。

字符串按位取反的本质是逐字节异或 0xFF
在 c++ 中,std::String 本质是 char 序列,而 char 在大多数平台是 8 位有符号类型(范围 -128 ~ 127)。按位取反(~)作用于 char 时,会先整型提升为 int,再取反,结果可能超出 char 范围,直接赋值回 char 会导致实现定义行为(如符号截断)。正确做法是:对每个字节显式与 0xFF 异或,等价于无符号意义上的按位取反。
- 错误写法:
ch = ~ch→ 可能产生负值、扩展问题、不可移植 - 推荐写法:
ch = ch ^ 0xFF或static_cast(ch) ^ 0xFF - 若需保持
char类型兼容性,强制转成unsigned char再运算,避免符号扩展干扰
使用 for 循环原地翻转每个字符的字节
这是最直观、可控性最强的方式,适用于所有 C++ 标准(C++11 起均可),且便于调试和加条件逻辑(如跳过非 ASCII 字符)。
std::string s = "ABC"; for (char& c : s) { c = static_cast(static_cast(c) ^ 0xFF); } // s 变为 "xffxffxff" 对应的字节序列(十六进制)
-
static_cast消除符号扩展歧义(c) -
^ 0xFF确保低 8 位全翻转,高位清零 -
static_cast转回(...) char类型以存入 string - 注意:结果可能含不可见控制字符(如
x00、x7f),打印时会“消失”或乱码,用printf("%02x ", (unsigned char)c)查看真实字节
用 std::transform 实现函数式风格处理
适合已有成熟转换逻辑、或想复用算法的场景。相比手写循环更紧凑,但可读性略低,且需注意 Lambda 捕获和类型转换细节。
std::string s = "Hello"; std::transform(s.begin(), s.end(), s.begin(), [](char c) { return static_cast(static_cast(c) ^ 0xFF); });
- 必须显式两次
static_cast:先升到unsigned char避免符号问题,再降回char匹配 string 元素类型 - 不能用
~c替代 —— 它返回int,隐式转char时可能截断出错 - 如果目标是生成新字符串而非原地修改,把第三个参数换成
std::back_inserter(new_s)
注意空字符 和二进制安全问题
std::string 是二进制安全的,支持嵌入