三分钟带你认识位图、手动实现位图

目录

1、概念

2、位图的简单使用

3、自己实现位图: 

4、位图的应用


1、概念

        位图,就是用每一位来存放某种状态,适用于海量数据【内存利用率高】整数,数据无重复的场景。通常是用来判断某个数据存不存在的

        说点好理解的,位图其实就是存在着一个映射关系【哈希】,原本存一个整数,需要4个字节,而在位图中,每一个比特位来标识一个数字,那此时,4个字节就有32个比特位,就是标识32个数字,当该数字出现过,该比特位为1,未出现过,该比特位为1

举例:

怎么确定位置:n/8确定区间,n%8确定该区间内所对应的下标 


2、位图的简单使用

上代码:

import java.util.BitSet;

public class Test {
    public static void main(String[] args) {
        BitSet bitSet = new BitSet();
        int[] arr = {3,6,8,25,34,45,37,58,26,47,2,5,1};
        //将数组中的数据放入位图
        for(int i = 0;i< arr.length;i++) {
            bitSet.set(arr[i]);
        }
        System.out.println("是否存在25:"+ bitSet.get(25));
        System.out.println("是否存在50: "+ bitSet.get(50));
    }
}

结果:


3、自己实现位图: 

 大致实现:


public class MyBitSet {
    public byte[] elem;
    public int usedSize;
    public MyBitSet() {
        this.elem = new byte[1];//默认1个字节,8个比特位
    }
    public MyBitSet(int n) {
        //为什么+1?假设n为12,那就是n/8等于1,还有4个没有给到,
        // 原本分配1个字节,8个比特位不够,+1后分配2个字节,16个比特位,够够了
        this.elem = new byte[n/8+1];
    }

    /**
     * 设置某一位为1
     * @param val
     */
    public void set(int val) {

    }

    /**
     * 判断当前位是不是1
     * @param val
     * @return
     */
    public boolean get(int val) {

    }

    /**
     * 设置当前位为0
     * @param val
     */
    public void reSet(int val) {

    }
}

接下来将里面的三个函数进行具体的实现:

第一个,置1:

思路: 

 

代码:

 

    /**
     * 设置某一位为1
     * @param val
     */
    public void set(int val) {
        if(val<0) {
            throw new IndexOutOfBoundsException();
        }
        int arrIndex = val/8;
        int bitIndex = val%8;

        elem[arrIndex] |= (1 << bitIndex);
        usedSize++;
    }

 第二个,判断:

思路 :

代码:

    /**
     * 判断当前位是不是1
     * @param val
     * @return
     */
    public boolean get(int val) {
        if(val<0) {
            throw new IndexOutOfBoundsException();
        }
        int arrIndex = val/8;
        int bitIndex = val%8;

        if((elem[arrIndex] & (1 << bitIndex)) != 0) {
            return true;
        }
        return false;
    }

 第三个,置0:

思路:

代码:

    /**
     * 设置当前位为0
     * @param val
     */
    public void reSet(int val) {
        if(val<0) {
            throw new IndexOutOfBoundsException();
        }
        int arrIndex = val/8;
        int bitIndex = val%8;
        if(!get(val)) {
            return;
        }
        elem[arrIndex] &= ~(1 << bitIndex);
        usedSize--;
    }

4、位图的应用

  • 快速查找某个数据是否在一个集合中
  • 排序 + 去重【按序取出即可】
  • 求两个集合的交集、并集等
  • 操作系统中磁盘块标记

好啦,这期就到这里了,我们下期再见!!! 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

龙洋静

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值