Java中Bitmap的实现
说bitmap之前,我们要明白数字在内存中的表示,如果说byte用8个二进制位表示,即可以表示个数,每个byte占8位,即每个byte占8行,在内存中这样形象的表示:
而bitmap结构,充分利用了每一行所有的位数,它将每个位置作为一个数,那么一行就可以模拟表示出8个数。
Bitmap介绍
bitmap是很有用的结构。所谓的bitmap就是用一个bit位来标记某个元素,而数组下标是该元素。
bitmap优势
bitmap经常用在大数据的题中,比如10亿个int类型的数,如果用int数组存储的话,那么需要大约4G内存,浪费内存。如果用bitmap解决,就比较方便。bitmap可以用int来模拟,也可以用byte来模拟,它只是逻辑上的概念,在java语言中写不出来,我们采用byte。一个byte占8个bit,如果每一个bit的值是有或者没有,即1或0,则如下图所示:
bitmap代码实现
第一步:构建特定长度的byte数组(new byte[capacity/8 + 1]),其中capacity为整数数组长度(如:10亿个数字等)
byte[] bits = new byte[getIndex(n) + 1];
第二步:计算数字num在byte[]中的位置(num/8和num >> 3一样),也就是说num在byte[k],算这个k是几
/** * num/8得到byte[]的index * @param num * @return */ public int getIndex(int num){ return num >> 3; }
第三步:计算数字num在byte[index]中的位置,就是在byte[index]的第几位,每个byte有8位(num % 8)
/** * num%8得到在byte[index]的位置 * @param num * @return */ public int getPosition(int num){ return num & 0x07; }
第四步:将所在位置从0变成1,其它位置不变
/** * 标记指定数字(num)在bitmap中的值,标记其已经出现过 * 将1左移position后,那个位置自然就是1,然后和以前的数据做|,这样,那个位置就替换成1了 * @param bits * @param num */ public void add(byte[] bits, int num){ bits[getIndex(num)] |= 1 << getPosition(num); }
解释如下图:
第五步:判断指定数字num是否存在
/** * 判断指定数字num是否存在<br/> * 将1左移position后,那个位置自然就是1,然后和以前的数据做&,判断是否为0即可 * @param bits * @param num * @return */ public boolean contains(byte[] bits, int num){ return (bits[getIndex(num)] & 1 << getPosition(num)) != 0; }
第六步:重置某一数字对应在bitmap中的值
/** * 重置某一数字对应在bitmap中的值<br/> * 对1进行左移,然后取反,最后与byte[index]作与操作。 * @param bits * @param num */ public void clear(byte[] bits, int num){ bits[getIndex(num)] &= ~(1 << getPosition(num)); }
全部代码如下:
public class Test { /** * 创建bitmap数组 */ public byte[] create(int n){ byte[] bits = new byte[getIndex(n) + 1]; for(int i = 0; i < n; i++){ add(bits, i); } System.out.println(contains(bits, 11)); int index = 1; for(byte bit : bits){ System.out.println("-------" + index++ + "-------"); showByte(bit); } return bits; } /** * 标记指定数字(num)在bitmap中的值,标记其已经出现过<br/> * 将1左移position后,那个位置自然就是1,然后和以前的数据做|,这样,那个位置就替换成1了 * @param bits * @param num */ public void add(byte[] bits, int num){ bits[getIndex(num)] |= 1 << getPosition(num); } /** * 判断指定数字num是否存在<br/> * 将1左移position后,那个位置自然就是1,然后和以前的数据做&,判断是否为0即可 * @param bits * @param num * @return */ public boolean contains(byte[] bits, int num){ return (bits[getIndex(num)] & 1 << getPosition(num)) != 0; } /** * num/8得到byte[]的index * @param num * @return */ public int getIndex(int num){ return num >> 3; } /** * num%8得到在byte[index]的位置 * @param num * @return */ public int getPosition(int num){ return num & 0x07; } /** * 重置某一数字对应在bitmap中的值<br/> * 对1进行左移,然后取反,最后与byte[index]作与操作。 * @param bits * @param num */ public void clear(byte[] bits, int num){ bits[getIndex(num)] &= ~(1 << getPosition(num)); } /** * 打印byte类型的变量<br/> * 将byte转换为一个长度为8的byte数组,数组每个值代表bit */ public void showByte(byte b){ byte[] array = new byte[8]; for(int i = 7; i >= 0; i--){ array[i] = (byte)(b & 1); b = (byte)(b >> 1); } for (byte b1 : array) { System.out.print(b1); System.out.print(" "); } System.out.println(); } public static void main(String[] args) { int n = 100; new Test().create(n); } }
结果如图,一共100个数:
杂谈
下面是我搜集的一些关于bitmap的,有时间再看。。。
https://blog.csdn.net/a3192048/article/details/80261699
这是关于java中原生的bitmap的实现,BitSet
https://blog.csdn.net/lushuaiyin/article/details/7546144
https://blog.csdn.net/y999666/article/details/51220833
https://blog.csdn.net/xia744510124/article/details/51509285/

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
预训练模型迁移学习
摘要:如何快速简便地解决图像分类问题呢?本文通过使用Keras及一个预训练模型的实例,教你如何通过迁移学习来解决这个问题。 深度学习正在迅速成为人工智能应用开发的主要工具。在计算机视觉、自然语言处理和语音识别等领域都已有成功的案例。 深度学习擅长解决的一个问题是图像分类。图像分类的目标是根据一组合理的类别对指定的图片进行分类。从深度学习的角度来看,图像分类问题可以通过迁移学习的方法来解决。 本文介绍了如何通过迁移学习来解决图像分类的问题。本文中所提出的实现方式是基于Python语言的Keras。 本文结构: 1)迁移学习 2)卷积神经网络 3)预训练模型的复用 4)迁移学习过程 5)深度卷积神经网络上的分类器 6)示例 7)总结 1、迁移学习 迁移学习在计算机视觉领域中是一种很流行的方法,因为它可以建立精确的模型,耗时更短。利用迁移学习,不是从零开始学习
- 下一篇
Netty Channel源码分析
原文:https://wangwei.one/posts/netty-channel-source-analyse.html 前面,我们大致了解了Netty中的几个核心组件。今天我们就来先来介绍Netty的网络通信组件,用于执行网络I/O操作 —— Channel。 Netty版本:4.1.30 概述 数据在网络中总是以字节的形式进行流通。我们在进行网络编程时选用何种传输方式编码(OIO、NIO等)决定了这些字节的传输方式。 在没有Netty之前,为了提升系统的并发能力,从OIO切换到NIO时,需要对代码进行大量的重构,因为相应的Java NIO 与 IO API大不相同。而Netty在这些Java原生API的基础上做了一层封装,对用户提供了高度抽象而又统一的API,从而让传输方式的切换不在变得困难,只需要直接使用即可,而不需要对整个代码进行重构。 Netty Channel UML netty channel族如下: 整个族群中,AbstractChannel 是最为关键的一个抽象类,从它继承出了AbstractNioChannel、AbstractOioChannel、Abstra...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- CentOS7设置SWAP分区,小内存服务器的救世主
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题
- CentOS7,8上快速安装Gitea,搭建Git服务器
- CentOS7编译安装Gcc9.2.0,解决mysql等软件编译问题
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- Windows10,CentOS7,CentOS8安装Nodejs环境
- CentOS7,CentOS8安装Elasticsearch6.8.6
- CentOS8安装Docker,最新的服务器搭配容器使用
- 设置Eclipse缩进为4个空格,增强代码规范