位运算符(&、|、^)对整数二进制位独立运算,常用于标志位检查(如access & fileaccess.read)、设置(|=)、翻转(^=);需注意优先级、负数补码、类型提升及与逻辑运算符区别。

位运算符在 C# 中的语法和基本行为
与(&)、或(|)、异或(^)在 C# 中是二元操作符,对整数类型的每一位独立运算。它们不关心数值含义,只看二进制表示——比如 5 & 3 实际计算的是 101 & 011 → 001,结果为 1。
注意:这些操作符有重载版本,但对 int、uint、long 等内置整型直接生效;对 bool 也支持(此时是逻辑运算,非位运算),别混淆。
常见错误现象:
- 把 && 写成 & 用于布尔判断,导致意外的位运算而非短路逻辑
- 对负数做位运算时没意识到补码表示,比如 -1 | 0xFF 结果仍是 -1(因为 -1 全是 1)
用 & 检查标志位是否开启(最常用场景)
这是位运算最典型的用途:把整数当一组开关用,每个 bit 表示一个布尔状态。例如 Windows API 或 .NET 的枚举常配合 [Flags] 特性:
[Flags]
enum FileAccess {
Read = 1, // 0001
Write = 2, // 0010
Execute = 4, // 0100
Delete = 8 // 1000
}判断某个权限是否存在,必须用 &,不能用 ==:
-
if ((access & FileAccess.Read) == FileAccess.Read)✅ 正确 -
if (access & FileAccess.Read)❌ 危险:非零即真,但若access是Read | Write(值为 3),该表达式仍为真,但语义模糊 -
if (access == FileAccess.Read)❌ 错误:忽略了组合权限
用 | 和 ^ 设置或翻转标志位
| 是“开启”某位,^ 是“翻转”某位,两者都常用于无副作用地修改权限集合:
- 添加权限:
access |= FileAccess.Write - 移除权限(需配合
~):access &= ~FileAccess.Execute—— 注意不是直接用^,否则会误翻其他位 - 切换权限(开变关、关变开):
access ^= FileAccess.Delete
性能影响:这些操作都是 CPU 原生指令,比方法调用或集合操作快得多,适合高频路径(如网络协议解析、图像像素处理)。
兼容性注意:byte、sbyte 参与位运算时会自动提升为 int,所以 (byte)5 & (byte)3 实际执行的是 5 & 3,结果是 int 类型,需要显式转换才能赋给 byte 变量。
容易被忽略的细节:运算符优先级和括号
位运算符优先级低于比较运算符(如 ==、!=),高于赋值(=),但远低于算术运算符。这导致很多隐蔽 bug:
-
if (x & 7 == 0)等价于if (x & (7 == 0))→ 永远是if (x & 0)→ 永远假 - 正确写法必须加括号:
if ((x & 7) == 0) - 复合赋值如
x ^= y + 1等价于x = x ^ (y + 1),加法先算,没问题;但x = x ^ y + 1就不对,因为^优先级低于+
建议:只要涉及混合运算,一律加括号;长期习惯后,能避免 90% 的位运算相关逻辑错误。










