状态位运算工具类

相信大家有碰到过这样一种情况:需要计算用户当前处于哪种状态,从而做出相应的处理。这时可以采用二进制记录某个用户的多个状态,然后通过位运算,便可快速得出当前状态的结果或者得出当前用户处于哪种状态。

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]公众号

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值