相信大家有碰到过这样一种情况:需要计算用户当前处于哪种状态,从而做出相应的处理。这时可以采用二进制记录某个用户的多个状态,然后通过位运算,便可快速得出当前状态的结果或者得出当前用户处于哪种状态。
1、背景
在做黄金理财项目的时候,黄金用户是有多种状态,后端需要记录当前用户处于何种状态,然后返回给前端或其他接口,进而引导用户做出相应的处理动作。用户需要在多种状态之间进行流转,而且状态是不可逆的。
我们通过下图:2 的n-1次幂的二进制和十进制数的对应关系可以看出,从二进制低位开始的第n位始终为 1 ,其他都是 0 。
对应于后端,在进行位运算的时候,二进制位第一位1对应的十进制数为 1
,第二位对应的十进制数为 2
,第三位对应的十进制数为 4
,以此类推,通过上面的图也可以看出。
接着我们定义枚举类code值为换算后的十进制数,后面使用工具类进行位运算的时候,也是拿这个code值进行计算的。
2、位运算符
可以采用二进制记录某个用户的多个状态,每一位记录一种状态。然后通过位运算处理的结果进行判断:
0-不存在状态,1-存在状态(或者) false-不存在状态,true-存在状态
3、工具类代码
核心代码主要通过下面四个通用方法实现,可直接引入该工具类到你的项目中使用,代码如下:
/**
* 位状态工具类
*/
public class BitStatesUtil {
/**
* 获取下一次对应的状态值
*
* @param stepValue 原有step值: 0,1,3,7,15
* @param codeDescPair 状态枚举,上面定义的状态枚举
* @return integer
*/
public static <R> Integer getNextStateValue(Integer stepValue, CodeDescPair<Integer, R> codeDescPair) {
if (judgeCurrentState(stepValue, codeDescPair)) {
return stepValue;
}
return (stepValue | codeDescPair.getCode());
}
/**
* 根据当前值返回上一个状态值
* 例如:getPreviousState(getPreviousState(63,32),16)
*
* @param stepValue 原有step值: 0,1,3,7,15
* @param codeDescPair 状态枚举,上面定义的状态枚举
* @return integer
*/
public static <R> Integer getPreviousStateValue(Integer stepValue, CodeDescPair<Integer, R> codeDescPair) {
if (!judgeCurrentState(stepValue, codeDescPair)) {
return stepValue;
}
return (stepValue ^ codeDescPair.getCode());
}
/**
* 判断当前状态值是否到最后一个状态
*
* @param stepValue 原有step值: 0,1,3,7,15
* @param enumCodesSum 枚举状态code值总和
* @return true 已完成最终步骤,false 未完成最终步骤
*/
public static boolean isStateDone(Integer stepValue, Integer enumCodesSum) {
return stepValue.compareTo(enumCodesSum) >= 0;
}
/**
* 判断当前处于哪个状态
*
* @param stepValue 原有step值: 0,1,3,7,15
* @param codeDescPair 状态枚举,上面定义的状态枚举
* @return true 已完成当前步骤,false 未完成当前步骤
*/
public static <R> boolean judgeCurrentState(Integer stepValue, CodeDescPair<Integer, R> codeDescPair) {
return Objects.nonNull(stepValue) && ((stepValue & codeDescPair.getCode())) != 0;
}
public static void main(String[] args) {
System.out.println(getPreviousState(getPreviousState(63,32),16));
}
}
举例:通过四位二进制位表示 stepValue,如果全是0(0000):值为0;如果第一位开始是1(0001): 值为1;如果第二位是开始是1(0011): 表示值为3,以此类推。
更多优质技术分享,请关注下面👇👇👇[老马Hony]公众号