先说一个核心结论:用一个 int 字段承载最多 32 个独立开关状态——这是 Java 位运算状态管理中最高效、最轻量的方案,没有之一。内存占用从几十字节直接压缩到 4 字节,所有操作均为单条 CPU 指令,无需集合、不触发 GC,性能与空间都逼近理论极限。当然,高效归高效,使用时有几个关键细节必须严格遵守,否则容易引发隐蔽问题。
定义清晰、可维护的状态常量
每个状态必须对应唯一的二进制位,通过 1 << n 构造。避免硬编码数字或重复赋值,这是最基础的规范。
- 用左移明确位序,例如
public static final int IS_ACTIVE = 1 << 0;、public static final int HAS_PROFILE = 1 << 3; - 命名务必体现业务含义,使用
IS_VERIFIED而不要用BIT_5,否则后续排查与团队协作会非常困难 - 一旦状态超过 32 个,要么切换为
long(64 位),要么封装BitSet。但注意BitSet不具备原子性,且存在对象开销
四大核心操作,写法需规范
所有状态变更均围绕 |、&~、&、^ 展开,避免 if 判断或遍历。这样写不仅执行速度快,代码也更清晰。
- 开启状态:
flags |= IS_SUBSCRIBED; - 关闭状态:
flags &= ~HAS_NOTIFICATION;——特别提醒,~必须作用于无符号常量,否则高位可能溢出,造成静默失效 - 判断启用:
if ((flags & CAN_EDIT) != 0),括号不能省略,因为&的优先级低于!= - 切换状态(有则关、无则开):
flags ^= IS_DIRTY;
与枚举和工具类配合,提升工程性
纯数字易出错,建议用枚举封装语义,并配套提供静态工具方法。
- 定义枚举时指定值,例如
enum Status { ACTIVE(1 << 0), LOCKED(1 << 1), PENDING(1 << 2); - 封装通用方法:
public static boolean contains(int mask, Status s) { return (mask & s.value) != 0; } - 数据库存取统一使用
INT或BIGINT字段,避免字符串拼接,同时支持索引查询
避开常见坑点,保证健壮性
高效背后隐藏着静默失效的风险,以下几个细节必须反复检查。
- 始终使用无符号常量(如
1U或1L),避免负数导致高位被置 1 - 多线程写入同一字段时,不能仅靠
volatile,必须使用AtomicInteger的getAndOr/getAndAnd等原子方法 - 调试时借助
Integer.toBinaryString(flags)或十六进制打印,一眼就能看清哪些位生效 - 状态总数千万别超过 32(int)或 64(long),超限未处理会被静默截断,届时排查将非常头疼
