Java中的位图和布隆过滤器(如果想知道Java中有关位图和布隆过滤器的知识点,那么只看这一篇就足够了!)

news2024/11/24 8:30:46

        前言:在学习之前的数据结构的时候,我们使用的数据量都不是很大,但是在生活中,我们常常面临着要处理大量数据结果并得出最终结果,那么有没有什么数据结构可以帮助我们实现这样的功能呢?


✨✨✨这里是秋刀鱼不做梦的BLOG

✨✨✨想要了解更多内容可以访问我的主页秋刀鱼不做梦-CSDN博客

那么在开始学习处理大量数据的数据结构之前,先让我们看一下本文大致的讲解内容:

目录

1.位图

        (1)位图的概念

        (2)位图的实现

        (3)位图的应用

2.布隆过滤器

        (1)布隆过滤器的概念

        (2)布隆过滤器的实现

        (3)布隆过滤器的优缺点

        (4)布隆过滤器的应用

3.海量数据处理中的应用场景

        (1)位图在海量数据处理中的应用

        (2)布隆过滤器在海量数据处理中的应用


1.位图

        (1)位图的概念

        首先我们要学习的数据结构就是位图,那么读者会发问了?什么是位图呢?以下是位图的概念:

        位图是一种空间优化的数据结构,适合用于表示大量数据的存在性。其基本思想是将每个数据用二进制的“0”或“1”表示,“1”表示数据存在,“0”表示数据不存在。位图特别适合用于对海量整数数据进行存在性检查或排序操作。

如图:

        通过上述的描述,相信读者对位图有了一定的了解,在了解了位图的定义之后,让我们用一个实例再来说明一下位图的作用:

        假设有40亿个不重复的无符号整数,我们需要判断某个数是否存在于这40亿个数中。通常的解决方法可能是将数据存储在数组或列表中,然后进行遍历或使用二分查找。然而,这两种方式的时间复杂度较高,而位图通过将每个整数映射到相应的比特位,能以较低的空间消耗实现高效查询。如上例,假设数据集中有40亿个整数,使用位图只需要约512MB的空间即可完成数据存储和查询。

        至此,我们就对位图有了初步的了解了!

        (2)位图的实现

        当我们了解了位图是什么之后,现在让我们自我实现一下位图。

——位图的核心是将整数映射到一个数组的特定位置并通过位操作来设置或获取该位置的值。下面是位图实现的代码:

public class MyBitSet {

    private byte[] elem;
    private int usedSize;

    // 构造函数,初始化位图大小
    public MyBitSet(int n) {
        elem = new byte[n / 8 + 1]; // 位图大小:n个比特位
    }

    // 设置某一位为1,表示对应的值存在
    public void set(int val) {
        int arrayIndex = val / 8;
        int bitIndex = val % 8;
        elem[arrayIndex] |= (1 << bitIndex);
        usedSize++;
    }

    // 检查某个值是否存在
    public boolean get(int val) {
        int arrayIndex = val / 8;
        int bitIndex = val % 8;
        return (elem[arrayIndex] & (1 << bitIndex)) != 0;
    }

    // 重置某一位为0,表示对应的值不存在
    public void reSet(int val) {
        int arrayIndex = val / 8;
        int bitIndex = val % 8;
        elem[arrayIndex] &= ~(1 << bitIndex);
        usedSize--;
    }

    // 获取已使用的比特位数量
    public int getUsedSize() {
        return usedSize;
    }
}

        根据上边的注释,我们可以很好的理解如何去实现一个位图,不过在此我们还是对上述的代码进行解释:

  • elem 数组用于存储比特位信息,每个字节(8位)可存储8个整数的存在状态。

  • set() 方法通过位运算将某一位设置为1。

  • get() 方法通过按位与运算检查某位是否为1,从而判断对应的整数是否存在。

  • reSet() 方法则将某一位重置为0,表示数据不存在。

        这样我们就了解了如何去自我实现位图了!

        (3)位图的应用

        了解完了什么是位图和如何去实现一个位图之后,读者可以会发问到:位图除了可以处理大量数据之外,位图在生活中还有什么用处呢?

以下为位图的一些应用场景:

  1. 快速判断数据是否存在:例如,在大数据集群中判断某个用户ID是否存在。
  2. 数据去重:通过位图可以快速判断某数据是否已经存在,从而避免重复操作。
  3. 集合运算:位图能够高效地实现集合的并集、交集等操作,尤其适合用于处理大规模的集合数据。
  4. 排序和去重:位图可以用于对大量整数进行排序和去重操作,例如,系统中磁盘块的标记操作。

        这样我们就了解了位图的应用了!

2.布隆过滤器

       同位图一样,在正式开始学习布隆过滤器之前,先让我们了解一下什么是布隆过滤器

        (1)布隆过滤器的概念

        布隆过滤器是一种概率型的数据结构,主要用于集合查询。与位图不同,布隆过滤器能够高效地判断某个元素是否“可能存在”或“肯定不存在”。它的特点是通过多个哈希函数将元素映射到位数组的不同位置,并设置对应比特位为1。当查询一个元素时,如果所有哈希函数映射的比特位都为1,则认为该元素“可能存在”;若有一个比特位为0,则该元素“肯定不存在”。

如图:

        光看上述的文字解释可能有点晦涩难懂,那么这里我们使用一个生活中的案例来对其进行解释:

        想象你在家里开派对,邀请了50位朋友,但你不记得是否某位朋友已经到达。你可以通过查看他们是否签了名(签名簿)来判断。每个朋友到达后,你会要求他们签名在一个特定的位置上,这类似于哈希函数的映射。如果你检查到所有他们应签的位都已经被签了,那就可能意味着他们已经来过了,但也可能是误判。

        相信通过上述的解释之后,读者可以对布隆过滤器有了更好的理解!

        (2)布隆过滤器的实现

        了解完了布隆过滤器的定义之后,现在让我们自我实现一个布隆过滤器。

        布隆过滤器的基本实现包含一个位数组和多个不同的哈希函数。每个哈希函数将输入的元素映射到位数组的一个位置,并将该位置的比特位设置为1。

下面是实现布隆过滤器的代码:

import java.util.BitSet;

// 简单哈希类
class SimpleHash {
    private int cap; // 哈希表的大小
    private int seed; // 哈希种子

    // 构造函数,初始化哈希表大小和种子
    public SimpleHash(int cap, int seed) {
        this.cap = cap;
        this.seed = seed;
    }

    // 哈希函数
    public int hash(String value) {
        int result = 0;
        int len = value.length();
        // 遍历字符串的每一个字符
        for (int i = 0; i < len; i++) {
            result = seed * result + value.charAt(i); // 计算哈希值
        }
        // 返回哈希值与 cap-1 的位与操作结果
        return (cap - 1) & result;
    }
}

// 布隆过滤器类
public class BloomFilter {
    private static final int DEFAULT_SIZE = 1 << 24; // 位数组大小
    private static final int[] seeds = {5, 7, 11, 13, 31, 37, 61}; // 哈希种子

    private BitSet bits = new BitSet(DEFAULT_SIZE); // 位数组
    private SimpleHash[] func = new SimpleHash[seeds.length]; // 哈希函数数组

    // 构造函数,初始化哈希函数
    public BloomFilter() {
        for (int i = 0; i < seeds.length; i++) {
            func[i] = new SimpleHash(DEFAULT_SIZE, seeds[i]); // 根据种子初始化 SimpleHash
        }
    }

    // 插入数据
    public void set(String value) {
        for (SimpleHash f : func) {
            bits.set(f.hash(value)); // 使用每个哈希函数设置位
        }
    }

    // 查询数据是否存在
    public boolean contains(String value) {
        for (SimpleHash f : func) {
            if (!bits.get(f.hash(value))) {
                return false; // 只要有一个哈希函数返回 false,就说明不在集合中
            }
        }
        return true; // 有可能误判
    }

    // 主函数,用于测试
    public static void main(String[] args) {
        BloomFilter filter = new BloomFilter();
        filter.set("test"); // 插入字符串 "test"
        System.out.println(filter.contains("test")); // 输出: true,表示 "test" 存在
        System.out.println(filter.contains("hello")); // 输出: false,表示 "hello" 不存在
    }
}

代码解释:

  • SimpleHash 类实现了哈希函数,通过不同的种子(seed)保证多个哈希函数的多样性。

  • BloomFilter 类使用位数组和多个哈希函数实现数据的插入与查询操作。set() 方法将数据插入布隆过滤器,而 contains() 方法则检查数据是否可能存在。

  • 布隆过滤器的查询可能存在误判,但能保证元素“肯定不存在”的准确性。

这样我们就大致的了解了如何去自我实现一个布隆过滤器了!

        (3)布隆过滤器的优缺点

        从上述的布隆过滤器的自我实现中,我们就可以发现布隆过滤器的一些优点和缺点,那么布隆过滤器有哪些优缺点呢?

优点

  1. 高效的空间利用:布隆过滤器能够通过少量的空间判断数据是否存在,大大减少了内存占用。

  2. 快速查询:插入和查询的时间复杂度为O(K),其中K为哈希函数的个数,与数据量大小无关。

  3. 并行化处理:多个哈希函数的操作可以并行执行,提高了性能。

缺点

  1. 误判:布隆过滤器可能会误判某些不存在的元素为存在,但它能保证不存在的元素一定不会被误判为存在。

  2. 不支持删除:一旦元素被插入布隆过滤器,无法删除,因为删除操作可能会误删除其他哈希值相同的元素。

        (4)布隆过滤器的应用

        通过上述对布隆过滤器的解释,我相信读者可以对布隆过滤器有一定的了解了,那么布隆过滤器有哪些应用呢?

布隆过滤器在以下场景中被广泛使用:

  1. 网络爬虫的URL去重:在大规模的网络爬虫中,布隆过滤器用于判断某个URL是否已经被访问过,防止重复爬取。

  2. 垃圾邮件过滤:布隆过滤器用于记录垃圾邮件发送者的地址,并在邮件服务器上快速过滤垃圾邮件。

  3. 数据库缓存穿透:在大规模分布式系统中,布隆过滤器可以用于防止频繁的数据库查询,减少缓存穿透的风险。

        这样我们就了解了布隆过滤器的应用场景了!

3.海量数据处理中的应用场景

        从上面的文章内容中我们可以知道,位图和布隆过滤器都是对海量数据进行处理的数据结构,那么其在海量数据处理中的应用场景有哪些呢?

        (1)位图在海量数据处理中的应用

        位图可以处理非常庞大的数据集,特别是在内存有限的情况下,位图通过其空间效率优势可以快速解决诸如数据去重、集合运算等问题。例如,假设有两个包含100亿个整数的文件,我们只有1GB的内存,可以通过位图来实现这两个集合的交集、并集等操作。

案例:找到只出现一次的整数

// 假设我们有一个包含100亿个整数的文件,下面的代码实现了位图的简单应用
MyBitSet bitSet = new MyBitSet(10000000000);  // 10 billion bits
for (int num : nums) {
    if (!bitSet.get(num)) {
        bitSet.set(num);  // 如果没有出现过,设置为1
    } else {
        bitSet.reSet(num);  // 如果再次出现,重置为0
    }
}
// 遍历 bitSet,输出所有只出现一次的整数

        (2)布隆过滤器在海量数据处理中的应用

        布隆过滤器由于其空间优势和高效的查询能力,在海量数据处理中被广泛应用于去重、快速查询等场景。例如,在网络爬虫中,布隆过滤器用于检测某个URL是否已经访问过,防止重复抓取。

案例:布隆过滤器在URL去重中的应用

BloomFilter filter = new BloomFilter();
for (String url : urls) {
    if (!filter.contains(url)) {
        // 处理未访问过的 URL
        filter.set(url);
    }
}

——通过上述的简单案例,希望读者可以对其在海量数据处理中的应用场景有一定的了解。


以上就是本篇文章的全部内容了~~~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2164359.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Circular dependency between the following tasks(gradle循环依赖的问题)

简介 目前公司项目主要使用gradle而不是maven&#xff0c;所以对gradle的使用不是很清楚&#xff0c;遇到这个问题的时候一直在晚上查资料&#xff0c;但是解决方案基本都是安卓的&#xff0c;后续先前辈请教了一下&#xff0c;才找到解决方案。 问题解析 本质上就是两个模块…

【QT 5 调试软件+Linux下调用脚本shell-无法调度+目录拼写+无法找目录+sudo权限(2)+问题解决方式+后续补充】

【QT 5 调试软件Linux下调用脚本shell-无法调度目录拼写无法找目录sudo权限&#xff08;2&#xff09;问题解决方式后续补充】 1、前言2、问题综述&#xff1a;自研qt上位机无法调度脚本&#xff08;1&#xff09;可能原因1&#xff1a;无法找到目录情况说明&#xff1a;解决思…

程序人生:软件测试 非技术性面试题【建议每个测试人观看】

1、自我介绍&#xff1a;三分钟左右 2、为什么从郑州/太原离职&#xff1f; 3、你的职业规划是什么样的&#xff1f; 4、对下一家公司有什么自己的想法吗&#xff1f; 5、你觉得作为一名测试工程师&#xff0c;应该具备什么样的素养&#xff1f; 6、你觉得管理层&#xff…

HTML | 外部引入 CSS 的2种方式:link和@import有什么区别?

外部引入 CSS 有2种方式&#xff0c;link 和 import。就结论而言&#xff0c;强烈建议使用 link &#xff0c;慎用 import 方式。 两者都是外部引用 CSS 的方式&#xff0c;但是存在一定的区别&#xff1a; &#xff08;1&#xff09;从属关系区别 link是HTML / XHTML标签&a…

react crash course 2024(3) jsx语法及组件

创建组件的文件结构 rafce创建组件内容 const NavBar () > {return (<div>NavBar</div>) }export default NavBar 使用该组件 组件传值 或者把props解构 设置默认值 5.用自定义组件包裹一些元素 把他们传值 元素直接变成了参数被传进来

求推荐网站建设公司?2024网站建设公司哪家好TOP3

好的网站建设公司在这个信息时代寻找起来会比较困难&#xff0c;因为你在任何一个搜索引擎搜索“网站建设”这个关键词&#xff0c;平台会给你推送成千上万家的建站公司&#xff0c;让人看得眼花缭乱&#xff0c;但究竟哪个更好&#xff0c;让人捉摸不透。 前段时间&#xff0…

多城联动、多形式开展网安周公益活动,开源网安传播网络安全知识

9月9日至15日&#xff0c;以“网络安全为人民&#xff0c;网络安全靠人民”为主题的2024年国家网络安全宣传周将在全国范围内统一开展&#xff0c;通过多样的形式、丰富的内容&#xff0c;助力全社会网络安全意识和防护技能提升。开源网安今年继续为各地企业、群众带来了丰富的…

Qt QFileDialog使用方法

头文件 #include <QFileDialog> 成员名称返回值说明getExistingDirectoryQString返回用户选中的文件夹路径getExistingDirectoryUrlQUrl与QFileDialog::getExistingDirectory()的主要区别来自于为用户提供的选择远程目录的能力getOpenFileNameQString返回用户选中的文件…

从更底层的角度理解网站的访问过程

文章目录 1.示例&#xff0c;访问www.baidu.com是如何返回数据的1.输入www.baidu.com回车2.检查本机的C:\Windows\System32\drivers\etc\hosts配置文件夹下有没有这个域名对应的映射&#xff1a; 1.示例&#xff0c;访问www.baidu.com是如何返回数据的 1.输入www.baidu.com回车…

18.2 k8s-apiserver监控源码解读

本节重点介绍 : k8s代码库和模块地址 下载 apiserver源码 apiserver中监控源码阅读 k8s源码地址分布 k8s代码库 访问github上k8s仓库&#xff0c;readme中给出了k8s 模块的代码地址举例图片 组件仓库列表 地址 Repositories currently staged here: k8s.io/apik8s.io/a…

Maven配置、下载教程(非常详细)

maven下载地址 Maven – 发行说明 – Maven 3.6.1 (apache.org) 1.配置settings.xml 下载完maven之后&#xff0c;保存在电脑中并解压 打开maven文件 -->conf-->settings.xml 使用记事本方式打开 打开之后找到这个地方&#xff0c;在电脑中创建一个文件夹(repository)…

起重机防摇摆技术如何达标-武汉正向科技

武汉正向科技防摇摆控制器 主要技术参数 1、防摇摆精度&#xff1a; 0.4 2、行车到达目标位置偏差位置偏差&#xff1a; 25mm 3、通讯方式&#xff1a;PROFINET / PROFIBUS / RS232 / RS422 / RS485&#xff1b; 4、消除载荷的摇摆达 96% 以上&#xff1b; 5、技术先进…

图像面积计算一般方法及MATLAB实现

一、引言 在数字图像处理中&#xff0c;经常需要获取感兴趣区域的面积属性&#xff0c;下面给出图像处理的一般步骤。 1.读入的彩色图像 2.将彩色图像转化为灰度图像 3.灰度图像转化为二值图像 4.区域标记 5.对每个区域的面积进行计算和显示 二、程序代码 %面积计算 cle…

- 串口通信

USART串口通信 目录 USART串口通信 回顾 USART串口通信 1、通信分类与作用 2、串口通信的相关参数&#xff08;重点&#xff09; 3、位协议层 -- RS232协议 4、STM32F103 中的串口外设 5、调试串口编程 -- &#xff08;1&#xff09;串口初始化&#xff1a;时钟、IO、…

【Git入门】使用 Git 进行项目管理:Word Count 程序开发与托管

在软件开发过程中&#xff0c;版本控制工具是不可或缺的。Git 作为一款强大的分布式版本控制工具&#xff0c;为开发者提供了高效的代码管理和协作方式。本博客将介绍如何下载安装 Git 版本管理工具&#xff0c;并使用 Git 和 GitHub 平台进行一个名为 Word Count 的项目开发与…

【鸿蒙HarmonyOS NEXT】数据存储之关系型数据库RDS

【鸿蒙HarmonyOS NEXT】数据存储之关系型数据库RDS 一、环境说明二、关系型数据库RDS介绍三、示例代码加以说明四、小结 一、环境说明 DevEco Studio 版本&#xff1a; API版本&#xff1a;以12为主 二、关系型数据库RDS介绍 1. RDS关系型数据库简介&#xff1a; 关系型数…

VMware提供虚拟硬盘并使得Oracle Linux集群共享块设备并绑定raw设备。

一、Vmware操作 nodeA : 1、创建SCSI控制器:类型为物理。 添加新磁盘。 类型为独立-持久。 nodeB: 新增磁盘,但是选择node A刚才创建的磁盘。 类型:独立-持久。 二、OS层操作 两台都要做绑定。 详细步骤 1. 创建 raw 设备节点 首先,确保 /dev/raw 目录存在。如果不…

5分钟精通Excel在go中的使用

一些简单操作可以在官方文档中找到&#xff0c;应该足够无经验的朋友们入门 介绍 - 《Excelize v2.2 中文文档》 - 书栈网 BookStack 这里贴一个中文版的链接&#xff08;以excelize库为例&#xff0c;相对其他库来说&#xff0c;体验很不错&#xff09;&#xff0c;不过要注…

沉浸式艺术创作:FLUX.1模型下的Java开发者体验之旅

文章目录 1 FLUX.1革命2 应用落地3 部署ComfyUI4 部署FLUX.15 导入工作流6 新的挑战 1 FLUX.1革命 FLUX.1&#xff1a;图像生成的新纪元 在人工智能的图像生成领域&#xff0c;FLUX.1模型的出现标志着一个新的时代。 由黑森林实验室&#xff08;Black Forest Labs&#xff0…

Android 车载应用开发指南 - CarService 详解(下)

车载应用正在改变人们的出行体验。从导航到娱乐、从安全到信息服务&#xff0c;车载应用的开发已成为汽车智能化发展的重要组成部分。而对于开发者来说&#xff0c;如何将自己的应用程序无缝集成到车载系统中&#xff0c;利用汽车的硬件和服务能力&#xff0c;是一个极具挑战性…