Java基础之哈希表与红黑树

news2025/1/19 17:20:37

文章目录

    • 一、哈希表
      • 1.1 JDK1.7版本之前哈希表(数组+链表,头插法)
      • 1.2 JDK1.8版本之后哈希表(数组+链表+红黑树,尾插法)
    • 二、红黑树
      • 2.1 使红黑树再次满足红黑规则
        • 2.1.1 使红黑树满足红黑规则方法一
        • 2.1.2 使红黑树满足红黑规则方法二
    • 三、JDK1.7与JDK1.8哈希表结构不同原因

一、哈希表

哈希表,又被称之为散列表(Hash Table),它是一种数据结构。在Java中:JDK1.7版本之前,哈希表的底层实现是数组+链表,即数据结构中所说的拉链法而在JDK1.8版本之后,哈希表的底层实现则是数组+链表+红黑树。而无论是JDK1.8之前还是JDK1.8之后,哈希表数组的初始值均为16,默认加载因子为0.75。

那什么是加载因子呢?例如数组的长度为16,存入数组的数据元素个数为4个,那么此数组的加载因子为4/16=0.25,而哈希表中规定的加载因子是为扩容作准备的。当哈希表的数组中存入数据达到16*0.75=12个时,数组会自动扩容,扩容至原来数组长度的两倍,即16*2=32,之后同样的若哈希表的数组中存入数据达到32*0.75=24个时,数组会自动扩容至原来数组长度的两倍,即32*2=64,之后以此类推。

那么哈希表具体是怎么实现的呢?接下来我们将结合例图来详细讲解。

1.1 JDK1.7版本之前哈希表(数组+链表,头插法)

当我们创建了一个HashSet或者HashMap对象时,系统会自动为我们创建一个初始值为16,加载因子为0.75的哈希表,如下图所示:

请添加图片描述


当我们向集合中添加一个数据时,系统会根据特定的散列函数来计算此数据元素将要存储在数组的哪一位置。例如若我们要添加一个"aa",经过系统散列函数计算后,得到散列值为4,即系统将会把"aa"存入到数组下标为4的位置,而此时数组为空,可以直接存入数据,如下图所示:

请添加图片描述


之后我们再次添加一个元素"aaa",若此时它的散列值也为4,系统判断出数组下标为4的位置已经存在元素,则系统将会采用头插法(旧元素插在新元素之后)将"aaa"插入数组,如下图所示:

请添加图片描述


之后我们再次添加一个元素"bbb",若此时它的散列值也为4,系统判断出数组下标为4的位置已经存在元素,则系统将会采用头插法(旧元素插在新元素之后)将"bbb"插入数组,如下图所示:
请添加图片描述


之后我们再次添加一个元素"b",若此时它的散列值为6,系统判断出数组下标为6的位置为空,则系统将会直接存入数据,如下图所示:
请添加图片描述


之后我们依次添加元素,直到数组的加载因子为0.75,即16*0.75=12,(注意:加载因子的计算我们只看数组上的元素,而不考虑链接的元素)如下图所示:
请添加图片描述


那么数组将会自动扩容至原来容量的2倍,之后以此类推。如下图所示:
请添加图片描述


1.2 JDK1.8版本之后哈希表(数组+链表+红黑树,尾插法)

当我们创建了一个HashSet或者HashMap对象时,系统会自动为我们创建一个初始值为16,加载因子为0.75的哈希表,如下图所示:
请添加图片描述


当我们向集合中添加一个数据时,系统会根据特定的散列函数来计算此数据元素将要存储在数组的哪一位置。例如若我们要添加一个"aa",经过系统散列函数计算后,得到散列值为4,即系统将会把"aa"存入到数组下标为4的位置,而此时数组为空,可以直接存入数据,如下图所示:
请添加图片描述


之后我们再次添加一个元素"aaa",若此时它的散列值也为4,系统判断出数组下标为4的位置已经存在元素,则系统将会采用尾插法(新元素插在旧元素之后)将"aaa"插入数组,如下图所示:
请添加图片描述


之后我们再次添加一个元素"bbb",若此时它的散列值也为4,系统判断出数组下标为4的位置已经存在元素,则系统将会采用尾插法(新元素插在旧元素之后)将"bbb"插入数组,如下图所示:
请添加图片描述


之后我们再次添加一个元素"b",若此时它的散列值为6,系统判断出数组下标为6的位置为空,则系统将会直接存入数据,如下图所示:
请添加图片描述


之后我们依次添加元素,直到数组的加载因子为0.75,即16*0.75=12,(注意:加载因子的计算我们只看数组上的元素,而不考虑链接的元素)如下图所示:
请添加图片描述


那么数组将会自动扩容至原来容量的2倍,之后以此类推。如下图所示:
请添加图片描述


二、红黑树

那么红黑树又是什么呢?

红黑树是一种自平衡(根据红黑规则定义的平衡,它和平衡二叉树定义的平衡不一样)的二叉查找树,它是一种数据结构,我们可以将它看作是一种特殊的二叉查找树,它的每一个节点都有存储位来表示结点的颜色,每个节点要么是黑色,要么是红色。

红黑规则:
1、每一个节点或是红色的,或是黑色的。
2、根节点必须是黑色的。
3、所有的叶子节点(叶子节点为空,我们可以理解为它们不是真实的节点)均是黑色的。
4、如果一个节点是红色的,那么它的父节点和孩子节点均不能为红色(即红色节点不能直接相连)
5、对于每一个节点,它到后代任意叶子节点的简单路径上,均包含相同数量的黑色节点。

那么红黑树具体是怎么实现的呢?接下来我们将结合例图来详细讲解。

假设我们当前有三个节点,分别为2,13,1并且节点颜色均为红色(注意,在创建红黑树时,节点颜色均为红色或黑色),如下图所示:
在这里插入图片描述


我们拿取节点2作为根节点,由于根节点必须为黑色,因此节点2需要变为黑色,之后为它添加两个叶子节点(叶子节点必须是黑色的),如下图所示:
在这里插入图片描述


之后我们向红黑树插入节点13,由于13>2,因此需要将节点13插入节点2的右子树,之后为它们配齐叶子节点,我们发现它满足红黑规则,如下图所示:

首先来看节点2,它作为根节点为黑色,并且它到后代任意叶子节点的简单路径上均只包含1个黑色节点。

之后我们来看节点13,它为红色节点,未与其他红色节点直接相连,并且它到后代任意叶子节点的简单路径上均只包含1个黑色节点。因此它满足红黑规则。

在这里插入图片描述


之后我们向红黑树插入节点1,由于1<2,因此需要将节点1插入节点2的左子树,之后为它们配齐叶子节点,同样我们发现它满足红黑规则,如下图所示:

首先来看节点2,它作为根节点为黑色,并且它到后代任意叶子节点的简单路径上均只包含1个黑色节点。

之后我们来看节点13,它为红色节点,未与其他红色节点直接相连,并且它到后代任意叶子节点的简单路径上均只包含1个黑色节点。

最后我们来看节点1,它为红色节点,未与其他红色节点直接相连,并且它到后代任意叶子节点的简单路径上均只包含1个黑色节点。

在这里插入图片描述


那如果我们再插入一个节点0,会发生什么呢?由于0<2,因此需要将节点0插入到节点2的左子树,由于0<1,因此需要将节点0插入到节点1的左子树。我们根据图解查看它会发生什么!!!

在这里插入图片描述


2.1 使红黑树再次满足红黑规则

它共有两种方式:

2.1.1 使红黑树满足红黑规则方法一

添加节点时,若节点的父亲节点和叔叔节点均为红色,则需要将父亲节点和叔叔节点均转变为黑色,并将祖父节点转换为红色,若祖父节点为根节点则需要再将其转变为黑色。

我们所写的例子即为此种情况,那么我们就需要将节点0的父节点(节点1)和叔叔节点(节点13)均转变为黑色,并将祖父节点(节点2)转变为红色,因为节点2为根节点,因此我们需要再次将其转变为黑色。如下图所示:
请添加图片描述


2.1.2 使红黑树满足红黑规则方法二

添加节点时,若节点的父亲节点为红色,叔叔节点为黑色,则需要将父亲节点转变为黑色,将祖父节点转换为红色,并以祖父节点为支点进行旋转。

之后我们再次添加一个节点-1,由于-1<2,因此需要将节点-1插入到节点2的左子树,由于-1<1,因此需要将节点-1插入到节点1的左子树,由于-1<0,需要将节点-1插入到节点0的左子树。如下图所示:
在这里插入图片描述


我们看到节点-1的父节点(节点0)为红色节点,叔叔节点(Nil)为黑色节点,此时需要将其父节点(节点0)转换为黑色,祖父节点(节点1)转换为红色,并以祖父节点(节点1)为支点进行旋转。如下图所示:
请添加图片描述


三、JDK1.7与JDK1.8哈希表结构不同原因

那为什么JDK1.8之后我们要再添加红黑树结构呢?我们一起来看一个例子:若此时数组的某一个链表长度为500个节点(极端例子,现实中不会出现),那么我们在查询某一个元素时,有时甚至需要查询500次才能查到我们想要查询的数据,但若是我们将链表转化为红黑树后,查询效率将会大大提高!!!因此我们就规定当数组元素达到64,链表长度达到8时,链表会自动转换为红黑树结构。

请添加图片描述


OK!!!至此哈希表我们就介绍完毕!!!

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

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

相关文章

JavaSE学习进阶day03_01 多态

第一章 多态 1.1 多态的形式 直接说什么是多态性太抽象了&#xff0c;我们先引入一个例子&#xff1a; 现在我定义了一个feed方法&#xff0c;在不同的类的对象调用这个方法时&#xff0c;都要改变形参&#xff0c;即每当我的对象不同时&#xff0c;都要重载该方法&#xff0…

【Java基础】day13

day13 一、Spring Bean 生命周期是怎样的&#xff1f; 详细过程分为以下几个步骤&#xff1a; ① 初始化 Bean 容器通过获取 BeanDefinition 中的信息进行实例化&#xff0c;这一步仅仅是简单的实例化&#xff0c;并没有进行依赖注入。 实例化的对象被包装在 BeanWrapper 对…

Qt音视频开发38-ffmpeg视频暂停录制的设计

一、前言 基本上各种播放器提供的录制视频接口&#xff0c;都是只有开始录制和结束录制两个&#xff0c;当然一般用的最多的也是这两个接口&#xff0c;但是实际使用过程中&#xff0c;还有一种可能需要中途暂停录制&#xff0c;暂停以后再次继续录制&#xff0c;将中间部分视…

RabbitMq架构设计原理

文章目录1、消息中间件1.1、什么是消息中间件1.2、传统的HTTP请求有什么缺点1.3、MQ的应用场景2、同步、多线程、以及MQ处理业务逻辑的区别2.1、同步发送Http 请求2.2、多线程处理业务逻辑2.3、MQ实现业务逻辑Mq和多线程之间的区别3、Mq消息中间件名词4、简单实现Mq的思路4.1、…

MySQL索引15连问,你能坚持到第几问?

目录 1.索引是什么? 2.MySQL索引有哪些类型 3.索引什么时候会失效? 4.哪些场景不适合建立索引? 5.为什么要用 B树&#xff0c;为什么不用二叉树? 6.一次B树索引树查找过程 7.什么是回表? 如何减少回表? 8.什么是覆盖索引? 9.聊聊索引的最左前缀原则 10.索引下…

Phind——一款面向开发人员的AI搜索引擎

目录前言一、Phind优点二、使用方法总结前言 Phind是一款面向开发人员的AI搜索引擎&#xff0c;它由大语言模型&#xff08;Large Language Model&#xff0c;LLM&#xff09;驱动 。相比于传统的搜索引擎&#xff0c;Phind有以下优势&#xff1a;自然语言搜索、面向开发者、AI…

【数据结构】期中考试一把梭(通宵版上)

前言 红中(Hong_zhong) CSDN内容合伙人、2023年新星计划web安全方向导师、 吉林师范大学网安大一的一名普通学生、摸鱼拿过大挑校二、 华为MindSpore截至目前最年轻的优秀开发者、IK&N战队队长、 阿里云专家博主、华为网络安全云享专家、腾讯云自媒体分享计划博主、 划了…

URL 和 HandlerMapping建立映射(11)

上一篇https://blog.csdn.net/chen_yao_kerr/article/details/130194864 我们已经分析了Spring MVC的配置&#xff0c;并且说明了如何通过注解的方式去替换各种各样的xml配置文件。本篇将更深入分析&#xff1a; 取代 springmvc.xml 配置 之前我们说过&#xff0c;定义一个类…

简述API(电商数据API)网关的概念和功能

API 网关 ( API gateway ) 前言 在 IOT &#xff08; 物联网 &#xff09;中&#xff0c;当我们的一些设备。例如&#xff08; 监控、传感器等 &#xff09;需要将收集到的数据和信息进行汇总时&#xff0c;我们就需要一个 API。&#xff08;如果你需要Taobao/JD/pinduoduo平台…

OpenAI-ChatGPT最新官方接口《语音智能转文本》全网最详细中英文实用指南和教程,助你零基础快速轻松掌握全新技术(六)(附源码)

Speech to text 语音智能转文本Introduction 导言Quickstart 快速开始Transcriptions 转录python代码cURL代码Translations 翻译python代码cURL代码Supported languages 支持的语言Longer inputs 长文件输入Prompting 提示其它资料下载Speech to text 语音转文本 Learn how to …

一句话设计模式11:过滤器模式

过滤器模式: 直接看 java8的filter; 文章目录过滤器模式: 直接看 java8的filter;前言一、过滤器模式的作用二、如何实现过滤器模式直接上代码总结前言 过滤器模式一般使用场景是: 过滤集合中的不同元素的一种手段,其实平时开发中你经常用,但是你不知道而已;(心里话: 这也算一种…

C++命名空间

C命名空间 C命名空间是一种用于组织代码的机制&#xff0c;它可以将全局命名空间划分为更小的、独立的部分&#xff0c;从而避免命名冲突和名字空间污染。在本文中&#xff0c;我们将介绍C命名空间的基本概念、使用方法和注意事项。 什么是命名空间&#xff1f; 命名空间是C…

QT

多平台C图形用户界面应用程序框架 集成了很多可以直接运用的图形的库 应用在windowns10系统 新建项目 有三种基类可以选择&#xff0c;开发是基于这三种基类的基础上&#xff0c;利用软件支持的QT语言进行界面元素添加与优化 代码添加&#xff08;添加代码时&#xff0c;大小…

flutter实战(1)-配置安装

目录支持的OS安装SDKwindows找到windows对应的SDK安装LINUXsnapd手动IDEMacLinux 或者 Windows 平台支持的OS 有以下这些OS可以安装配置flutter 安装SDK windows 要想安装和运行 Flutter&#xff0c;你的开发环境至少应该满足如下的需求&#xff1a; 操作系统&#xff1a;W…

组合预测模型 | SSA-LSTM、LSTM麻雀算法优化长短期记忆神经网络时间序列预测(Matlab程序)

组合预测模型 | SSA-LSTM、LSTM麻雀算法优化长短期记忆神经网络时间序列预测(Matlab程序) 目录 组合预测模型 | SSA-LSTM、LSTM麻雀算法优化长短期记忆神经网络时间序列预测(Matlab程序)预测结果评价指标基本介绍程序设计参考资料预测结果 评价指标 SSA-LSTM优化得到的最优…

[TIFS 2022] FLCert:可证明安全的联邦学习免受中毒攻击

FLCert: Provably Secure Federated Learning Against Poisoning Attacks | IEEE Journals & Magazine | IEEE Xplore 摘要 由于其分布式性质&#xff0c;联邦学习容易受到中毒攻击&#xff0c;其中恶意客户端通过操纵其本地训练数据和/或发送到云服务器的本地模型更新来毒…

阿里“通义千问”大模型上线!让生成式AI更贴近中国人生活

阿里版的 ChatGPT 语言大模型来了。 张勇在峰会上表示&#xff0c;阿里巴巴所有产品未来将接入“通义千问”大模型&#xff0c;进行全面改造。他认为&#xff0c;面向AI时代&#xff0c;所有产品都值得用大模型重新升级。 目前&#xff0c;钉钉、天猫精灵等产品已接入通义千问测…

PYQT5学习笔记00——Pycharm环境搭建以及配置项目虚拟环境教程

1、安装基本环境 需要的基本环境有python3.x的解释器、pip包管理工具以及pipenv虚拟环境管理工具。   我们安装了python后&#xff0c;pip包管理工具会自带安装&#xff0c;pipenv虚拟环境管理工具我们使用pycharm即可&#xff0c;无需使用python自带的。 python解释器下载地…

【Git代码仓库托管】上海道宁为您提供构建、扩展和交付安全软件的完整开发人员平台

GitHub是用于 构建、扩展和交付安全软件的 完整开发人员平台 通过提高开发人员速度的工具 推动创新 加快高质量软件开发 GitHub提供无限的存储库 一流的版本控制和 世界上强大的开源社区 因此您的团队可以 更高效地协同工作 开发商介绍 GitHub归属于微软公司&#xf…

Java EE企业级应用开发(SSM)第6章

第6章Spring MVC应用一.预习笔记 1.Spring MVC的请求参数 项目的基础配置 web.xml springmvc-config.xml jar包资源引入&#xff1a; 1-1&#xff1a;获取默认参数 jsp页面如下&#xff1a; Controller如下&#xff1a; 1-2&#xff1a;简单数据类型&#xff08;获取数据不…