很多人觉得位运算只能用来炫技、做算法题,实则不然。在真实业务、框架底层、权限系统、状态标记中,位掩码(BitMask) 是非常成熟、轻量化、高性能的实现方案。今天就带你落地:用 1 个 int 整数,存储多种组合状态、权限开关,替代冗余的布尔字段、枚举组合,代码更简洁、存储更节省。
一、什么是位掩码?
核心思想:利用二进制每一位 0 / 1代表一种独立状态 / 权限。
一个 int 占 32 位,理论上:单个 int 可以存储 32 种独立开关状态。
相比方案:
多个 Boolean 字段:冗余、数据库字段多、判断繁琐 List 集合存权限:占用内存、判断包含效率低 枚举多条件并列:代码臃肿,if 嵌套爆炸
位掩码优势:
极省存储:1 个数字 = 多种状态 操作高效:位运算底层指令,速度极快 组合灵活:自由叠加、移除、判断状态 大量开源框架、中间件、权限系统原生使用
二、定义基础状态常量
以用户功能权限举例,每一位代表一项独立权限:
/** * 位掩码 - 权限标记常量 * 每一个常量,单独占用二进制中的一位 */ public class PermissionMask { // 第0位:查看权限 public static final int VIEW = 1 << 0; // 0001 // 第1位:新增权限 public static final int ADD = 1 << 1; // 0010 // 第2位:编辑权限 public static final int EDIT = 1 << 2; // 0100 // 第3位:删除权限 public static final int DELETE = 1 << 3; // 1000 }1 << n:1 左移 n 位,保证每个权限独占一位,互不干扰、可以自由组合。
三、四个位运算操作
1. 叠加权限:位或
// 初始无任何权限 int userMask = 0; // 赋予:查看 + 新增 userMask = userMask | PermissionMask.VIEW; userMask = userMask | PermissionMask.ADD;原理:
两个二进制位进行 | 运算:只要有一个为 1,结果就是 1;否则为 0。
userMask 初始: 00000000 VIEW: 00000001 ---------------------------- userMask | VIEW:00000001 (即赋权:查看)2. 判断是否拥有权限:位与 &
判断当前是否包含某项权限,使用 位与。
// 判断是否有编辑权限 public static boolean hasEdit(int mask) { return (mask & PermissionMask.EDIT) != 0; }原理:如果相对应位都是1,则结果为1,否则为0
00000101 (拥有第0位、3位权限,即拥查看、编辑权限) & 00000100 (EDIT) ------------------- 00000100 (根据位与计算,如有则回因为相对应位都是1,则结果为1 ,再判断不等于0,即表示拥有权限)3. 移除指定权限:按位取反 + 位与
精准取消某一项权限,不影响其他已开启状态。
// 移除删除权限 userMask = userMask & ~PermissionMask.DELETE;原理:
DELETE = 1 << 3 = 00001000(第 3 位为 1,从 0 开始数) 假如当前 userMask = 00001101(拥有权限:第 0 位、2 位、 3 位,即拥有查看、编辑、删除权限)
//取反 ~DELETE DELETE = 00001000 ~DELETE = 11110111 (第 3 位变成 0,其余位变成 1) //位与操作 userMask & ~DELETE userMask = 00001101 ~DELETE = 11110111 ---------------------- & 结果 = 00000101(第三位变为了0,即取消掉了)4. 权限反转:异或 ^
已有则关闭、没有则开启,适合开关类状态。
// 反转编辑权限 userMask = userMask ^ PermissionMask.EDIT;原理:
假设 EDIT = 1 << 2 = 00000100 当前 userMask = 00000101(第 0 位和第 2 位为 1,即拥有查看、编辑权限)
// 执行 userMask ^ EDIT userMask = 00000101 EDIT = 00000100 ---------------------- ^ 结果 = 00000001(第二位反转了,即编辑权限反转了)四、完整可运行工具类
public class PermissionUtil { /** * 追加权限 */ public static int addFlag(int mask, int flag) { return mask | flag; } /** * 移除权限 */ public static int removeFlag(int mask, int flag) { return mask & ~flag; } /** * 判断是否包含指定权限 */ public static boolean hasFlag(int mask, int flag) { return (mask & flag) != 0; } /** * 反转状态/权限 */ public static int toggleFlag(int mask, int flag) { return mask ^ flag; } }六、真实业务适用场景
系统权限控制
菜单权限、操作按钮权限、功能白名单组合
多状态标记
工单状态、任务标签、设备多种运行模式
配置项开关
功能全局开关、规则组合配置、多策略组合
框架底层源码
Spring、MyBatis、JDK 大量使用位掩码存储配置与状态
七、使用规范与避坑
统一用 1 << n 定义常量
避免手动写死数字,可读性差、后期难以维护。
禁止超过位数限制
int 最多 32 位、long 64 位,状态极多改用集合 / 配置表。另外要注意若使用第31位(1<<31)需注意符号位带来的调试/展示影响
不要过度滥用
简单 1~2 个开关,直接用 boolean 更直观;多状态组合场景,才是位掩码的主场。
数据库存储友好
直接存一个 int 数字即可,不用加多个字段,拓展性极强。
八、总结
位掩码才是位运算在业务中真正有价值的用法,不是奇技淫巧;位或 | 加状态、位与 & 判断状态、取反 + 位与删状态、异或 ^ 做开关;用一个整型承载多种组合状态,精简代码、节省存储、高效判断;代码可读性、性能、拓展性三者平衡。