0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

对比Java中HashMap的一些异同

科技绿洲 来源:Java技术指北 作者:Java技术指北 2023-09-30 10:46 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

1.前言

我们回顾一下之前讲到的Redis的字典结构,示意图如下:

图片

Redis的字典本质上来说也是数组+链表的数据结构,这与Java中HashMap的数据结构很类似。

由上述结构示意图也能看出,字典dict中维护了一个ht数组,而且只有两个元素,这两个元素是其扩容的关键点,这个我们后面会讲到。

Redis中的哈希对象在以下条件时,使用ziplist编码,

  • 哈希对象保存的所有键值的字符串长度都小于64字节
  • 哈希对象保存的键值对数量小于512个。

否则哈希对象会使用hashtable编码, 而hashtable则时使用了字典作为底层实现的。

如下redis 哈希对象编码由ziplist 变成hashtable

图片

2.增加元素与键冲突

当不同的键值经过哈希算法与散列算法之后被分配到了同一个哈希表数组的同一个索引上,那么这之后就会有键冲突。

Redis 哈希表解决哈希冲突同样是使用了链表地址法。使用哈希节点的next指针来链接同一个哈希表数组索引上的元素。不过Redis会将新添加的哈希节点加入到链表的表头位置。

如下所示:如果程序要将键值对 (k2 , v2 ) 添加到如下的哈希表中,而且计算的书的索引为1,那么和 (k1 v1) 将产生冲突。解决冲突时,会将两个节点使用next指针链接起来。而且会将新节点添加到链表表头的位置。

图片
哈希表1

图片
链表解决hash冲突之后的哈希表

3.rehash 扩容过程

哈希表不断的增加元素,其元素数量达到一定的比例之后,程序会对哈希表进行相应的扩展。通过执行rehash (重新散列)操作完成操作。其步骤如下:

  • 执行扩展操作时会将字典中的ht[1] 哈希表大小设置成 第一个大于等于 ht[0] 的 ht[0].used * 2 的 2^n (2的n次幂)
  • 将保存早ht[0] 中的所有的键值对 rehash到ht[1] 上, rehash过程中会重新计算哈希值和索引值。
  • 当ht[0]中所有的键值对都迁移到ht[1]上时,释放ht[0], 并将ht[1] 设置成 ht[0], 并在ht[1]上建一个空的哈希表。

将下图中的字典做rehash操作:

图片

  1. ht[0].used 是4,4*2 = 8 ,2的3次方8 是第一个大于4 的 2的n次幂。即程序会将ht[1] 的大小设置成8 ,并分配空间,结构示意如下:

图片

  1. 将ht[0] 上的几个键值对全部都rehash到ht[1] 上面,如下图:

图片

  1. 释放ht[0],并将ht[1] 设置成 ht[0] , 然后为ht[1]分配一个空白的哈希表 如下图:

图片

以上是一个rehash的过程示意。

4.渐进式rehash

上面讲的是一个rehash的理论过程,redis实际操作时并不会一次将所有的迁移一次性完成。

如果键值对数量非常庞大,那么迁移过程必然需要花费一点时间。由此可知,服务器也不可能一次将所有的键值对迁移,需要分多次,逐渐将ht[0] 里面的键值对迁移到ht[1]中,

其步骤如下:

  • 首先会给ht[1]分配内存空间,此时redis字典拥有两个哈希表
  • 字典中维护一个rehashidx的计数器,将其值设置为0,表示rehash工作开始
  • 在rehash期间,程序依然可以进行增删改查的操作,除此之外还会顺带将ht[0]上 rehashidx索引上所有的键值对rehash到ht[1]上,rehash的工作完成后会将rehashidx的值加1
  • 随着字典的操作,ht[0]上的所有键值全部都rehash到ht[1]上时,程序会将rehashidx的值设为-1 ,表示rehash操作已经完成

在渐进式rehash的过程中,redis字典依然是可以进行增删改查的操作, 其中增加元素的时候会将元素直接保存到ht[1]中, 而删除,查找,更新的操作会在两个哈希表中进行, 查找时会先在ht[0]中进行查找,然后会在ht[1]中进行查找。以上措施可以保证ht[0]中的元素只会减少,最终变成空表。

总结

Redis字典和Java中的HashMap的相似点和不同。

相似之处:

  1. 键值对存储:Redis 字典和 Java 的 HashMap 都是键值对存储的数据结构,它们可以通过键来快速查找对应的值。
  2. 高效的查找:Redis 字典和 Java 的 HashMap 都使用了哈希表来实现,因此在查找操作上具有高效性能。
  3. Redis 字典使用的时哈希表作为底层,并且每个字典维护了两个哈希表,ht[0] 时主要使用的哈希表,而ht[1] 是在rehash过程是才会使用到的表。
  4. 哈希表的底层同样是使用了数组 + 链表的结构, 与Java 中HashMap 相似,只不过Java8 以后增加了红黑树,在特定情况下会替换链表。

不同之处:

  1. 哈希表增加元素遇到哈希冲突是会将新添加的元素放到链表头,而Java HashMap会将其放到链表尾,
  2. 扩容过程中redis的字典是渐进式扩容,扩容期间还是可以进行操作的,而Java的HashMap扩容需要一次性完成。
  3. 存储方式:Redis 字典是一种基于内存的数据结构,用于在内存中存储键值对。而 Java 的 HashMap 可以在内存中存储,也可以持久化到磁盘上。
  4. 分布式支持:Redis 是一种分布式数据库,可以在多台服务器上进行数据共享和存储。而 Java 的 HashMap 只能在单个 JVM 中使用。
  5. 数据类型:Redis 字典可以存储多种数据类型,如字符串、列表、集合等,而 Java 的 HashMap 只能存储对象类型。
  6. 持久化:Redis 字典可以将数据持久化到磁盘上,以便在重启后恢复数据。而 Java 的 HashMap 需要自己实现数据的序列化和反序列化来实现持久化。
  7. 并发性:Redis 字典是线程安全的,可以支持多个客户端并发访问。而 Java 的 HashMap 在多线程环境下需要进行额外的同步处理才能保证线程安全。
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • JAVA
    +关注

    关注

    20

    文章

    2997

    浏览量

    115685
  • 编码
    +关注

    关注

    6

    文章

    1015

    浏览量

    56657
  • 字符串
    +关注

    关注

    1

    文章

    594

    浏览量

    23044
  • 数据结构
    +关注

    关注

    3

    文章

    573

    浏览量

    41374
  • hashmap
    +关注

    关注

    0

    文章

    15

    浏览量

    2507
收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    java集合干货系列

    而不是数组  数组的长度固定,集合长度可变  数组只能通过下标访问元素,类型固定,而有的集合可以通过任意类型查找所映射的具体对象。  整理的集合框架思维导图  个人整理的Java集合框架思维导图,动态维护。导出的图片无法查看备注的一些信息,所以需要源文件的童鞋可以回复邮箱
    发表于 12-14 15:11

    关于java性能优化的一些细节

    代码优化 ,个很重要的课题。可能有些人觉得没用,一些细小的地方有什么好修改的,改与不改对于代码的运行效率有什么影响呢?这个问题我是这么考虑的,就像大海里面的鲸鱼样,它吃条小虾米有
    发表于 10-11 09:23

    Java-4 Java的类

    关于JAVA入门的一些简明教程,还有一些简单的程序,适合入门PPT。
    发表于 04-29 11:28 0次下载

    JAVA关于this和that的一些知识

    新手在入门 Java 的过程定会踩很多关于 this 的坑,出现问题的本质就是 this 指针的指向和自己想的不样。笔者在入门学习的过程
    发表于 09-25 14:55 0次下载

    关于Java HashMap的认知

    HashMap详解 HashMap 和 HashSet 是 Java Collection Framework 的两个重要成员,其中 HashMap 是 Map 接口的常用实现类,Ha
    发表于 09-27 16:34 0次下载
    关于<b class='flag-5'>Java</b> <b class='flag-5'>HashMap</b>的认知

    java异常处理设计和一些建议

    程序设计在程序设计,进行异常处理是非常关键和重要的部分。个程序的异常处理框架的好坏直接影响到整个项目的代码质量以及后期维护成本和难度。试想下,如果
    发表于 09-28 11:48 0次下载
    <b class='flag-5'>java</b>异常处理设计和<b class='flag-5'>一些</b>建议

    关于java一些基础知识解析

    j2ee 全称Java 2 Enterprise Edition,是Java种企业版,用于企业级应用开发。 j2se 全称Java 2 Standard Edi
    的头像 发表于 02-05 14:43 5157次阅读
    关于<b class='flag-5'>java</b>的<b class='flag-5'>一些</b>基础知识解析

    十个问题带你了解和掌握java HashMap

    本文档内容介绍了十个问题带你了解和掌握java HashMap及源代码,供参考
    发表于 03-12 15:41 0次下载

    java学习—探秘Java的String、StringBuilder以及StringBuffer

    探秘JavaString、StringBuilder以及StringBuffer 相信String这个类是Java中使用得最频繁的类之,并且又是各大公司面试喜欢问到的地方,今天就来
    发表于 03-13 10:58 0次下载

    Java一些基础面试题资料合集免费下载

    本文档的主要内容详细介绍的是Java一些基础面试题资料合集免费下载。目录,1.Java基础知识篇 2.Java web基础知识总结 3.Java
    发表于 05-10 18:13 0次下载
    <b class='flag-5'>Java</b>的<b class='flag-5'>一些</b>基础面试题资料合集免费下载

    Java代码的一些技巧分享

    有时候我们为了统管理会把一些变量放到 yml 配置文件
    的头像 发表于 03-16 12:05 1776次阅读

    学习linux内核的一些建议

    学习linux内核,这个可不像学门语言,c或者java个月或者3月你就能精通掌握。学习linux内核是需要步循序渐进,掌握正确的l
    发表于 05-07 15:20 892次阅读
    学习linux内核的<b class='flag-5'>一些</b>建议

    介绍一些大功率IGBT模块应用一些技术

    PPT主要介绍了大功率IGBT模块应用一些技术,包括参数解读、器件选型、驱动技术、保护方法以及失效分析等。
    发表于 09-05 11:36 2509次阅读

    runtime 的一些对比选型和应用

    基于 io-uring 为 Rust 提供异步支持,并在此基础上研发通用网关。 本文包括以下内容: 介绍 Rust 异步 Runtime; Monoio 的一些设计精要; Runtime 对比选型与应用。 02 Rust
    的头像 发表于 05-26 15:48 1426次阅读
    runtime 的<b class='flag-5'>一些</b><b class='flag-5'>对比</b>选型和应用

    java中常用的包有哪些

    Java种面向对象的高级编程语言,它具有平台无关性和可扩展性。Java中有很多常用的包,这些包提供了丰富的类库和工具,用于开发各种类型的应用程序。下面是Java
    的头像 发表于 11-22 15:10 2253次阅读