[Java基础要义]HashMap、LinkedHashMap元素遍历机制探讨

2024-03-13 19:48

本文主要是介绍[Java基础要义]HashMap、LinkedHashMap元素遍历机制探讨,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 Map作为键值对Entry<K,V>的的容器,对其内部 键值对Entry<K,V> 的遍历总归是要有一个顺序的。

      本文重点讨论HashMap及其子类LinkedHashMap的遍历机制,总结出两者的特点和适用情况。

 

 CSDN-2014博客之星投票啦

CSDN-2014博客之星   评选开始啦,如果您觉得我的文章对您有所帮助,请您点击左边栏的图片投我一票,您的支持是我分享知识的强大动力!

 

 

 

1.HashMap的遍历机制

           HashMap 提供了两个遍历访问其内部元素Entry<k,v>的接口:

              1.       Set<Map.Entry<K,V>> entrySet()     返回此映射所包含的映射关系的 Set 视图。

             2.       Set<K> keySet()              返回此映射中所包含的键的 Set 视图。

     实际上,第二个借口表示的Key的顺序,和第一个接口返回的Entry顺序是对应的,也就是说:这两种接口对HashMap的元素遍历的顺序相相同的。  那么,HashMap遍历内部Entry<K,V> 的顺序是什么呢? 搞清楚这个问题,先要知道其内部结构是怎样的。          
 

           HashMap内部对键值对的存储结构使用的是数组+链表的形式。其结构如下图所示:

 

HashMap内部Entry<K,V>的遍历顺序:

          对Entry[] table 数组,从index=0开始,依次遍历table[i] 上的链表上的Entry对象。

 

由于HashMap在存储Entry对象的时候,是根据Key的hash值判定存储到Entry[] table数组的哪一个索引值表示的链表上,所以笼统地说就是:使用hashMap.put(Key key,Value value)会将 对应的Entry<Key,Value>对象随机地分配到某个Entry[] table数组的元素表示的链表上。换一句话说就是:

     对HashMap遍历Entry对象的顺序和Entry对象的存储顺序之间没有任何关系。

但是,我们有时候想要遍历HashMap的元素Entry的顺序和其存储的顺序一致,HashMap显然不能满足条件了。而LinkedHashMap则可以满足这个需要。

 

2. LinkedHashMap 的遍历机制

          LinkedHashMap 是HashMap的子类,它可以实现对容器内Entry的存储顺序和对Entry的遍历顺序保持一致。

       为了实现这个功能,LinkedHashMap内部使用了一个Entry类型的双向链表用这个双向链表记录Entry的存储顺序。当需要对该Map进行遍历的时候,实际上是遍历的是这个双向链表。

        LinkedHashMap内部使用的LinkedHashMap.Entry类继承自 Map.Ent ry类,在其基础上增加了LinkedHashMap.Entry类型的两个字段,用来引用该Entry在双向链表中的前面的Entry对象和后面的Entry对象。

       它的内部会在 Map.Entry 类的基础上,增加两个Entry类型的引用:before,after。LinkedHashMap使用一个双向连表,将其内部所有的Entry串起来。

 

            我们将通过以下例子,来了解内部双向链表是怎样构造的:

 

 
  1. LinkedHashMap linkedHashMap = new LinkedHashMap();

  2. linkedHashMap.put("name","louis");

  3. linkedHashMap.put("age","24");

  4. linkedHashMap.put("sex","male");

 

上述的代码除了会将对应的Entry对象放置到在Entry[] table 表示的数组链表中外,还会将该Entry对象添加到其内部维护的双向链表中。对应的LinkedHashMap内部的双向链表变化如下:

 

对LinkedHashMap进行遍历的策略:

           从 header.after 指向的Entry对象开始,然后一直沿着此链表 遍历下去,直到某个entry.after == header 为止,完成遍历。

 

由此,就可以保证遍历LinkedHashMap内元素的顺序,就是Entry插入到LinkedHashMap中的顺序。

将上面代码中定义的linkedHashMap 遍历输出,会发现遍历的顺序跟插入的顺序完全一致:

 

 
  1. Iterator<Map.Entry> iterator= linkedHashMap.entrySet().iterator();

  2.  
  3. while(iterator.hasNext())

  4. {

  5. Map.Entry entry = iterator.next();

  6. System.out.println(entry.getKey()+":"+entry.getValue());

  7. }

结果输出:

 

 

根据Entry<K,V>插入LinkedHashMap的顺序进行遍历的方式叫做:按插入顺序遍历

另外,LinkedHashMap还支持一种遍历顺序,叫做:Get读取顺序

     如果LinkedHashMap的这个Get读取遍历顺序开启,那么,当我们在LinkedHashMap上调用get(key) 方法时,会导致内部 key对应的Entry在双向链表中的位置移动到双向链表的最后。

比如,如果当前LinkedHashMap内部的双向链表的情况如下:

相关代码如下:

 

 
  1. //默认情况下LinkedHashMap的遍历模式是插入模式,如果想显式地指定为get读取模式,那么要将

  2. //其构造方法的参数置为true,(false 表示的是插入模式)

  3. LinkedHashMap linkedHashMap = new LinkedHashMap(16, (float) 0.75,true);

  4.  
  5. linkedHashMap.put("name","louis");

  6. linkedHashMap.put("age","24");

  7. linkedHashMap.put("sex","male");

  8. linkedHashMap.get("name");//get()方法调用,导致对应的entry移动到双向链表的最后位置

  9.  
  10. Iterator<Map.Entry> iterator= linkedHashMap.entrySet().iterator();

  11.  
  12. while(iterator.hasNext())

  13. {

  14. Map.Entry entry = iterator.next();

  15. System.out.println(entry.getKey()+":"+entry.getValue());

  16. }

 

 

3. 总结

1.HashMap对元素的遍历顺序跟Entry插入的顺序无关,而LinkedHashMap对元素的遍历顺序可以跟Entry<K,V>插入的顺序保持一致。

2.当LinkedHashMap处于Get获取顺序遍历模式下,当执行get() 操作时,会将对应的Entry<k,v>移到遍历的最后位置。

3.LinkedHashMap处于按插入顺序遍历的模式下,如果新插入的<key,value> 对应的key已经存在,对应的Entry在遍历顺序中的位置并不会改变。

4. 除了遍历顺序外,其他特性HashMap和LinkedHashMap基本相同。

 

转载地址 :https://louluan.blog.csdn.net/article/details/43017071

这篇关于[Java基础要义]HashMap、LinkedHashMap元素遍历机制探讨的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/806004

相关文章

Spring LDAP目录服务的使用示例

《SpringLDAP目录服务的使用示例》本文主要介绍了SpringLDAP目录服务的使用示例... 目录引言一、Spring LDAP基础二、LdapTemplate详解三、LDAP对象映射四、基本LDAP操作4.1 查询操作4.2 添加操作4.3 修改操作4.4 删除操作五、认证与授权六、高级特性与最佳

Spring Shell 命令行实现交互式Shell应用开发

《SpringShell命令行实现交互式Shell应用开发》本文主要介绍了SpringShell命令行实现交互式Shell应用开发,能够帮助开发者快速构建功能丰富的命令行应用程序,具有一定的参考价... 目录引言一、Spring Shell概述二、创建命令类三、命令参数处理四、命令分组与帮助系统五、自定义S

SpringKafka错误处理(重试机制与死信队列)

《SpringKafka错误处理(重试机制与死信队列)》SpringKafka提供了全面的错误处理机制,通过灵活的重试策略和死信队列处理,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、Spring Kafka错误处理基础二、配置重试机制三、死信队列实现四、特定异常的处理策略五

SpringSecurity JWT基于令牌的无状态认证实现

《SpringSecurityJWT基于令牌的无状态认证实现》SpringSecurity中实现基于JWT的无状态认证是一种常见的做法,本文就来介绍一下SpringSecurityJWT基于令牌的无... 目录引言一、JWT基本原理与结构二、Spring Security JWT依赖配置三、JWT令牌生成与

Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码

《Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码》:本文主要介绍Java中日期时间转换的多种方法,包括将Date转换为LocalD... 目录一、Date转LocalDateTime二、Date转LocalDate三、LocalDateTim

如何配置Spring Boot中的Jackson序列化

《如何配置SpringBoot中的Jackson序列化》在开发基于SpringBoot的应用程序时,Jackson是默认的JSON序列化和反序列化工具,本文将详细介绍如何在SpringBoot中配置... 目录配置Spring Boot中的Jackson序列化1. 为什么需要自定义Jackson配置?2.

Java中使用Hutool进行AES加密解密的方法举例

《Java中使用Hutool进行AES加密解密的方法举例》AES是一种对称加密,所谓对称加密就是加密与解密使用的秘钥是一个,下面:本文主要介绍Java中使用Hutool进行AES加密解密的相关资料... 目录前言一、Hutool简介与引入1.1 Hutool简介1.2 引入Hutool二、AES加密解密基础

mysql的基础语句和外键查询及其语句详解(推荐)

《mysql的基础语句和外键查询及其语句详解(推荐)》:本文主要介绍mysql的基础语句和外键查询及其语句详解(推荐),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋... 目录一、mysql 基础语句1. 数据库操作 创建数据库2. 表操作 创建表3. CRUD 操作二、外键

Spring Boot项目部署命令java -jar的各种参数及作用详解

《SpringBoot项目部署命令java-jar的各种参数及作用详解》:本文主要介绍SpringBoot项目部署命令java-jar的各种参数及作用的相关资料,包括设置内存大小、垃圾回收... 目录前言一、基础命令结构二、常见的 Java 命令参数1. 设置内存大小2. 配置垃圾回收器3. 配置线程栈大小

SpringBoot实现微信小程序支付功能

《SpringBoot实现微信小程序支付功能》小程序支付功能已成为众多应用的核心需求之一,本文主要介绍了SpringBoot实现微信小程序支付功能,文中通过示例代码介绍的非常详细,对大家的学习或者工作... 目录一、引言二、准备工作(一)微信支付商户平台配置(二)Spring Boot项目搭建(三)配置文件