位运算是Java中直接操作整数二进制位的运算方式,由于直接作用于底层二进制数据,运算速度比普通的加减乘除等算术运算更快,在需要高性能处理、状态标记、权限控制等场景中有着广泛的应用。Java提供了多种位运算符,每种运算符对应不同的二进制操作逻辑,下面我们逐一介绍其用法。

Java常见的位运算符分类
Java中的位运算符可以分为以下几类,分别对应不同的二进制操作逻辑:
- 按位与(&):两个二进制位都为1时结果才为1,否则为0
- 按位或(|):两个二进制位只要有一个为1结果就为1,都为0时结果才为0
- 按位异或(^):两个二进制位不同时结果为1,相同时结果为0
- 按位取反(~):对单个二进制位取反,1变0,0变1
- 左移(<<):将二进制位整体向左移动指定位数,右边补0
- 右移(>>):将二进制位整体向右移动指定位数,左边补符号位
- 无符号右移(>>>):将二进制位整体向右移动指定位数,左边补0
基础位运算符用法示例
按位与运算
按位与运算常用于屏蔽某些二进制位,比如获取一个数的低8位数据,示例代码如下:
public class BitAndDemo {
public static void main(String[] args) {
int a = 13; // 二进制为 00001101
int b = 7; // 二进制为 00000111
int result = a & b; // 按位与运算
// 计算过程:00001101 & 00000111 = 00000101,对应十进制5
System.out.println("13 & 7 的结果是:" + result);
}
}
按位或运算
按位或运算常用于将某些二进制位设置为1,比如给状态值添加某个标记,示例代码如下:
public class BitOrDemo {
public static void main(String[] args) {
int a = 8; // 二进制为 00001000
int b = 3; // 二进制为 00000011
int result = a | b; // 按位或运算
// 计算过程:00001000 | 00000011 = 00001011,对应十进制11
System.out.println("8 | 3 的结果是:" + result);
}
}
按位异或运算
按位异或运算有一个特性:一个数和自身异或结果为0,和0异或结果还是自身,常用于数据加密、交换变量值等场景,示例代码如下:
public class BitXorDemo {
public static void main(String[] args) {
int a = 5; // 二进制为 00000101
int b = 9; // 二进制为 00001001
int result = a ^ b; // 按位异或运算
// 计算过程:00000101 ^ 00001001 = 00001100,对应十进制12
System.out.println("5 ^ 9 的结果是:" + result);
// 异或交换两个变量的值,不需要临时变量
int x = 10;
int y = 20;
x = x ^ y;
y = x ^ y; // 此时等价于 (x^y)^y = x
x = x ^ y; // 此时等价于 (x^y)^x = y
System.out.println("交换后 x=" + x + ", y=" + y);
}
}
按位取反运算
按位取反运算会将整数的所有二进制位取反,需要注意Java中整数采用补码表示,取反后结果可能和直观预期不同,示例代码如下:
public class BitNotDemo {
public static void main(String[] args) {
int a = 5; // 二进制补码为 00000101
int result = ~a; // 按位取反
// 计算过程:~00000101 = 11111010,对应补码的十进制是-6
System.out.println("~5 的结果是:" + result);
}
}
移位运算符用法示例
左移运算
左移运算相当于给原数乘以2的n次方(在没有溢出的情况下),示例代码如下:
public class LeftShiftDemo {
public static void main(String[] args) {
int a = 3; // 二进制为 00000011
int result = a << 2; // 左移2位
// 计算过程:00000011 左移2位变成 00001100,对应十进制12,即3*2^2=12
System.out.println("3 << 2 的结果是:" + result);
}
}
右移运算
右移运算相当于给原数除以2的n次方(向下取整),正数右移左边补0,负数右移左边补1,示例代码如下:
public class RightShiftDemo {
public static void main(String[] args) {
int a = 12; // 二进制为 00001100
int result1 = a >> 2; // 右移2位
// 计算过程:00001100 右移2位变成 00000011,对应十进制3,即12/2^2=3
System.out.println("12 >> 2 的结果是:" + result1);
int b = -12; // 二进制补码为 11110100
int result2 = b >> 2; // 右移2位,左边补1
// 计算过程:11110100 右移2位变成 11111101,对应十进制-3
System.out.println("-12 >> 2 的结果是:" + result2);
}
}
无符号右移运算
无符号右移不管正负,左边都补0,所以负数无符号右移后会变成很大的正数,示例代码如下:
public class UnsignedRightShiftDemo {
public static void main(String[] args) {
int a = -12; // 二进制补码为 11111111 11111111 11111111 11110100
int result = a >>> 2; // 无符号右移2位,左边补0
System.out.println("-12 >>> 2 的结果是:" + result);
}
}
位运算常见应用场景
位运算在实际开发中有很多实用场景,比如:
- 状态标记:用一个整数的不同二进制位表示不同的状态,比如用第0位表示是否启用,第1位表示是否审核通过,通过位运算快速修改和判断状态
- 权限控制:每个权限对应一个二进制位,用户的权限值是多个权限位的或运算结果,判断是否有某个权限时用按位与运算即可
- 性能优化:在需要大量乘除2的操作时,用移位运算代替可以提升执行效率
- 哈希计算:很多哈希算法的实现中会用到位运算来打乱数据的分布,减少哈希冲突
需要注意的是,位运算虽然效率高,但可读性相对较差,如果不是对性能有极高要求或者场景确实需要,不建议过度使用位运算,避免降低代码的可维护性。