java使用二进制解决一个字段代表多个状态的问题

2020-03-31  本文已影响0人  jiezzy

https://www.jianshu.com/p/61bf95cfc9a7

package com.king.common.utils.binary;
 
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
 
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;
 
/**
 * 使用二进制解决一个字段代表多个状态的问题
 *
 * 有没有更好的方法,处理一条记录多个状态的问题呢?
 * 使用二进制位,每个位置,代表一个状态,多个位置代表多个状态,存储的时候转换为int类型存储。
 *
 * 以产品属性存储为例。
 * 热门:0000 0001
 * 新品:0000 0010
 * 特卖:0000 0100
 *
 * 多个状态的组合
 * 热门+新品:0000 0011
 * 热门+特卖:0000 0101
 * 热门+新品+特卖:0000 0111
 *
 * 存储时将二进制转换为int数值。
 *
 * 二进制位的查询
 * 以Oracle为例,使用bitadd函数,如:
 *     SELECT * FROM T_PRODUCT WHERE bitand(property,1)=1
 * 以Mysql为例,使用&运算符,如:
 *     SELECT data_code,data_value FROM `tb_ceshi_data` where ceshi_code='xxx' and data_value & 8
 * java中对二进制位的处理
 * 1)是否包含:&与运算符,比如:(1&3)==1
 * 2)组装:|或运算符,比如 1|2=3
 *
 * 如果再扩展,清仓状态,可以使用 0000 1000
 *
 * 二进制实现的好处
 * 以上内容讲解了使用二进制位来表示状态的方式,实现数据库设计和程序调用。有以下好处
 * 1)存储精简
 * 2)扩展性强
 *
 * 缺点:
 * 1)增加了理解复杂度;
 *
 * 以下是六个状态情况的二进制与十进制存储对照表:
 * 00000001 1  参数1
 * 00000010 2  参数2
 * 00000100 4  参数3
 * 00001000 8  参数4
 * 00010000 16 参数5
 * 00100000 32 参数6
 *
 * 00000011 3  参数1+参数2
 * 00000101 5  参数1+参数3
 * 00000110 6  参数2+参数3
 * 00000111 7  参数1+参数2+参数3
 * 00001001 9  参数1+参数4
 * 00001011 11 参数1+参数2+参数4
 * 00001101 13 参数1+参数3+参数4
 * 00001111 15 参数1+参数2+参数3+参数4
 * 00010110 22 参数2+参数3+参数5
 * 00111111 63 参数1+参数2+参数3+参数4+参数5+参数6
 *
 *---------------------------------------java中操作二进制的运算符总结(&,| , ^, ~, , >>> )-----------------------------------------
 * &运算符总结
 * 1.特点:二元操作符,操作两个二进制数据;两个二进制数最低位对齐,只有当两个对位数都是1时才为1,否则为0
 * 2.案例:
 *    int a = 3 & 2 ;
 *    System.out.println(a); //结果为 2
 * 3.分析:
 *    3的二进制补码表示为:
 *        00000000 00000000 00000000 00000011
 *    2的二进制补码表示为:
 *        00000000 00000000 00000000 00000010
 *    运算:3 & 2
 *        00000000 00000000 00000000 00000011
 *    &   00000000 00000000 00000000 00000010
 *    -------------------------------------------
 *        00000000 00000000 00000000 00000010            二进制是2
 *
 * @Description
 * @Author HHJ
 * @Date 2019-05-20 16:14
 */
public class BinaryUtil {
    private static final Logger log = LoggerFactory.getLogger(BinaryUtil.class);
 
    /**
     * 求list数值之和(list="1,2,4,8,16,32....")
     * (使用场景是前端将list传入进来.后端计算list之和,然后将结果直接入库)
     *
     * @param list
     * @return
     */
    public static Integer sum(List<Integer> list) {
        if (CollectionUtils.isEmpty(list)) {
            return 0;
        }
        return list.stream().reduce(Integer::sum).orElse(0);
    }
 
    /**
     * 二进制转换为十进制
     *(在这种方式下,该方法无用)
     * 
     * @param binaryString
     * @return
     */
    public static Integer binaryToInteger(String binaryString) {
        Integer i = Integer.parseInt(binaryString, 2);
        return i;
    }
 
    /**
     * 以base为底数.value为值,算对数.例如
     *     计算64以4为底的对数:
     *     double res = log(64, 4);//4*4*4=64
     *     Assert.isTrue(res==3)
     * 
     * @param value
     * @param base
     * @return
     */
    public static double log(double value, double base) {
        return Math.log(value) / Math.log(base);
    }
 
    /**
     * 十进制数值 转对应的list
     * (使用场景是将DB中对应的int转对应的list,然后将结果返回给前端)
     * 
     * @param data
     * @return
     */
    public static List<Integer> toList(Integer data) {
        List<Integer> list = new ArrayList<>();
        if (data <= 0) {
            return list;
        }
        double count = log((double) data, 2d);
        for (int i = 0; i < count + 1; i++) {
            int n = 1 << i;
            if ((data & n) == n) {
                list.add(n);
            }
        }
        return list;
    }
 
    public static void main(String[] args) {
        List<Integer> realList = toList(22);
        System.out.println(realList.toString());
        int integer = binaryToInteger("00001011");
        Assert.isTrue(11 == integer, "this expression must be true");
 
        List<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(4);
        list.add(8);
        Integer sum = sum(list);
        System.out.println("sum=" + sum);
        List<Integer> realList1 = toList(sum );
        System.out.println(realList1.toString());
        int data = 34;
        List<Integer> result = toList(data);
 
        System.out.println("result.size:" + result.size() + "...结果:" + result.toString());
    }
}
————————————————
版权声明:本文为CSDN博主「九龙冰室083」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/hhj13978064496/java/article/details/90379886
上一篇 下一篇

猜你喜欢

热点阅读