简单分析hashmap 中的entryset()方法
来源:互联网 发布:点o是什么文件 linux 编辑:程序博客网 时间:2024/06/07 19:11
关于Java的HashMap.entrySet(),文档是这样描述的:这个方法返回一个Set,这个Set是HashMap的视图,对Map的操作会在Set上反映出来,反过来也是。原文是
Returns a Set view of the mappings contained in this map. Theset is backed by the map, so changes to the map are reflected inthe set, and vice-versa.
本文通过源码简单分析这一功能的实现。
首先要简单介绍一下HashMap的内部存储。我们知道,Map是用来存储key-value类型数据的,一个对在Map的接口定义中被定义为Entry,HashMap内部实现了Entry接口。HashMap内部维护一个Entry数组。
transient Entry[] table;当put一个新元素的时候,根据key的hash值计算出对应的数组下标。数组的每个元素是一个链表的头指针,用来存储具有相同下标的Entry。
Entry[] table --- | 0 | -> entry_0_0 -> entry_0_1 -> null--- | 1 | -> null --- | | ... |n-1| -> entry_n-1_0 -> null---entrySet()方法返回的是一个特殊的Set,定义为HashMap的内部私有类
private final class EntrySet extends AbstractSet>主要看一下这个Set的iterator()方法。这个方法很简单,返回一个EntryIterator类型的实例。EntryIterator类型是泛型HashIterator的一个子类,这个类的内容很简单,唯一的代码是在next()函数中调用了HashIterator的nextEntry()方法。所以,重点就变成了分析nextEntry()方法。上述过程见下面的图示
HashMap |- table <------------------------------------\ \entrySet() |iterates | HashMap.HashIterator | |returns ^ \-nextEntry() V - ^HashMap.EntrySet | | \- iterator() |extends | | || | instantiats | |calls \----------> HashMap.EntryIterator | \-next() /HashIterator通过遍历table数组,实现对HashMap的遍历。内部维护几个变量:index记录当前在table数组中的下标,current用来记录当前在table[index]这个链表中的位置,next指向current的下一个元素。nextEntry()的完整代码如下:
final Entry nextEntry() { if (modCount != expectedModCount) thrownew ConcurrentModificationEx第一个if用来判断在多线程的情况下是否出现并发错误,这里暂时不讨论。如果next不是null,那么返回并更新next。更新方法是第三个if的内容:如果当前链表还没有结束,则简单的把next向后移一个;否则在table中查找下一个非空的slot。
总结一下,HashMap的entrySet()方法返回一个特殊的Set,这个Set使用EntryIterator遍历,而这个Iterator则直接操作于HashMap的内部存储结构table上。通过这种方式实现了“视图”的功能。整个过程不需要任何辅助存储空间。
p.s.从这一点也可以看出为什么entrySet()是遍历HashMap最高效的方法,原因很简单,因为这种方式和HashMap内部的存储方式是一致的。
- 简单分析hashmap 中的entryset()方法
- HashMap 源码分析 -- entrySet()
- HashMap的entrySet()方法
- HashMap 中的keySet()和entrySet()方法的比较
- HashMap中的keySet()和entrySet()
- 再来谈谈HashMap中的entrySet
- HashMap EntrySet源码 分析(修改版)
- HashMap之entrySet( )底层实现原理分析
- HashMap中的indexFor方法分析
- Java中HashMap两种遍历方法(keyset & entryset快)
- 遍历hashmap的键值二 ,通过 entrySet()方法
- java.util.map中的方法entrySet解析
- 使用entrySet遍历HashMap
- 使用entrySet遍历HashMap
- HashMap简单分析
- HashMap简单原理分析
- HashMap的entrySet与keySet
- HashMap的entrySet与keySet
- 内存和缓存的区别
- 使用cfdisk 新建一个分区
- 通知公告TextSwitcher自动上下滚动带点击事件
- Netty之有效规避内存泄漏
- 学习笔记-mybatis数据处理的四种方式
- 简单分析hashmap 中的entryset()方法
- oracle数据库启动报错ORA-12560问题解决方法
- MFC大神之路
- Vue.js结合vue-router和webpack编写单页路由项目
- R语言技巧:对读取常用的集中格式的介绍
- Java虚拟机工具之堆栈跟踪工具jstack检测对象wait方法
- 二分查找(BinarySearch)
- P_CalcPlan
- C# 将byte 保存为文件 转为文件