之前对Java Se中的线性表作了简单的说明。这一篇就来看看Map。

Map系列的类,并不是说所有的类都继承了Map接口,而是说他们的元素都是以<Key, Value>形式设计的。

Dictionary

这个类现在不推荐使用了,但也有必要说一下,在它的描述中,有这么一句:Any non-null object can be used as a key and as a value。

现在都改用Map接口了。

Map

这个接口用于替换Dictionary的。这种集合提供了三种视图方式:

1)a set of keys

2)collection of Values

3)set of Entry<key, value>

Hashtable

这个类继承了Dictionary实现了Map。下面就重点看看这个类如何实现。

从这里看Hashtable是一个数组,数组的元素是Entry。

在看看Entry的设计:

这个Entry是一个单链表。

所以Hashtable的数据结构就可以认为是这样的:

为什么会这样设计?

以为硬件的原因,支持数组和链表两种形式,所以其他的数据结构(集合)都是基于这两类基础结构实现的。

Hashtable的设计,关键一点是hash,它计算的是key的hash,通过key的hashcode来计算这个元素所在的单链表在数组中的索引,然后根据索引取到单链表,然后进行其他的操作。

接下来看看它的几个重要方法如何实现:

put

  1. public synchronized V put(K key, V value) {
  2.  
  3. // Make sure the value is not null
  4.  
  5. if (value == null) {
  6.  
  7. throw new NullPointerException();
  8.  
  9. }
  10.  
  11. // 如果key已经在这个hashtable中存在,就覆盖原有的value,并返回原有的value。
  12.  
  13. // 从这一小段代码中,我们就可以看出hashtable的结构确实如上面所想的那样。
  14.  
  15. Entry tab[] = table;
  16.  
  17. int hash = key.hashCode();
  18.  
  19. int index = (hash & 0x7FFFFFFF) % tab.length;
  20.  
  21. for (Entry<K,V> e = tab[index] ; e != null ; e = e.next) {
  22.  
  23. if ((e.hash == hash) && e.key.equals(key)) {
  24.  
  25. V old = e.value;
  26.  
  27. e.value = value;
  28.  
  29. return old;
  30.  
  31. }
  32.  
  33. }
  34.  
  35. modCount++;
  36.  
  37. if (count >= threshold) {
  38.  
  39. // Rehash the table if the threshold is exceeded
  40.  
  41. rehash();
  42.  
  43. tab = table;
  44.  
  45. index = (hash & 0x7FFFFFFF) % tab.length;
  46.  
  47. }
  48.  
  49. // 如果不存在,就把key value 封装成一个Entry,然后插入到相应位置
  50.  
  51. // 在之前的代码中已经计算出它的索引了,接下来的任务就是把entry插入到数组索引位置处的单链表里。插入的时候是从头部插入的。
  52.  
  53. Entry<K,V> e = tab[index];
  54.  
  55. tab[index] = new Entry<K,V>(hash, key, value, e);
  56.  
  57. count++;
  58.  
  59. return null;
  60.  
  61. }

Hashtable#put(key, value)

常用的方法get(key),remove(key)的方法就不分析了,因为没有必要,只要你了解了这个数据结构,自己就可以实现它的几个常用的方法。

keys

拿到的结果其实是一个枚举器(不过这个枚举器比较特殊,实现了Enumeration接口和Iterable接口),并不是一个集合。

同样的道理,elements也是如此,拿到的结果是一个遍历value的枚举器。

HashMap

HashMap的存储结构如下图所示,因为源码部分太多,所以就截了这么一张图,但也足够说明问题了:

从图上很容易看出,HashMap也是使用了数组实现的,数组中的元素是Node,再看看Node的设计:

Node也是一个单链表,看来HashMap与Hashtable在结构上是基本一致的。

也就是说他的模型也是下面图中所示:

如果说HashMap的结构只是这个样子的话,就没有必要存在了。为什么这么说呢,关键点还是在Node上,看看他的两个子类:

Entry:

每个节点又添加了一两个引用:before,after,这样一来结构就可以猜想为:

但其实并不是这个样子的,会根据hashcode进行一些算法处理,形成链式结构,也就是LinkedHashMap了。

如果有兴趣可以了解一些LinkedHashMap是如何实现的。

TreeNode:

每个节点都可以是一个树。就让结构变得更为复杂了。

Java Se :Map 系列的更多相关文章

  1. Java Se 基础系列(笔记) -- OO

    记录所学到的关于Java Se的一些基础知识 1.对象是通过“属性(成员变量)”和“方法”来分别对应事物所具有的静态属性和动态属性 2.类(Class)是对某一类事物的抽象,对象(Object)为某个 ...

  2. Java Se 基础系列(笔记) -- Exception && Array

    Exception 1.java 异常是java提供的用于处理程序中错误(指在程序运行的过程中发生的一些异常事件)的一种机制 2.java程序的执行过程中如果发生异常事件则自动生产一个异常类对象,该对 ...

  3. Java Se 基础系列(笔记) -- BasicDataType

    java.lang.String类代表不可变的字符序列 String类常用方法:1.public char charAt(int index); -- 返回下标为index的字符 2.public i ...

  4. Java SE开发系列-JDK下载安装

    JDK下载安装 JDK是Java的开发环境,目前JDK内部也包含了JRE,JRE主要是JAVA程序的运行环境. 点击官方下载地址,按着下图操作即可下载对应系统的不同版本JDK. 进入页面滑到页面底部点 ...

  5. 4.Java集合总结系列:Map接口及其实现

    一.Map接口 Map集合的特点是:通过key值找到对应的value值,key值是唯一的,value可以重复.Map中的元素是无序的,但是也有实现了排序的Map实现类,如:TreeMap. 上面Map ...

  6. Java SE之For增强与Iterator遍历器提取数据(附Map.Entry)

    增强for循环: 1.操作数组 2.操作List集合 3.操作Map集合    1.map.values()法    2.map.keySet()法  [传统方法]    3.Map.Entry法   ...

  7. java se系列(十二)集合

    1.集合 1.1.什么是集合 存储对象的容器,面向对象语言对事物的体现,都是以对象的形式来体现的,所以为了方便对多个对象的操作,存储对象,集合是存储对象最常用的一种方式.集合的出现就是为了持有对象.集 ...

  8. java se系列(一)开发前奏

    1. 软硬件知识 电子计算机:俗称电脑,是一种能够按照程序运行,自动.高速处理海量数据的现代化智能电子设备.由硬件和软件所组成,没有安装任何软件的计算机称为裸机 cpu:是一台计算机的运算核心和控制核 ...

  9. Java SE 9 新增特性

    Java SE 9 新增特性 作者:Grey 原文地址: Java SE 9 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new_ ...

随机推荐

  1. 2016暑假多校联合---Death Sequence(递推、前向星)

    原题链接 Problem Description You may heard of the Joseph Problem, the story comes from a Jewish historia ...

  2. NYOJ 737---石子归并(GarsiaWachs算法)

    原题链接 描述    有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子并成为一堆.合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这两堆石子的和,经过N-1次合并后成为一堆.求 ...

  3. 由select * from table where 1=1中的1=1说开数据库

    众多网站都有select * from table where 1=1此类语句的介绍,并且,针对该类语句,讲得实在是让人越看越迷茫(一个抄袭一个的,简直不像话),不知道是在说什么,导致很多新手不得要领 ...

  4. HTML5开发IDE介绍

    开发html5的工具可供选择的有:Intellij Idea.Eclipse.WebStorm三个软件.在PC端与手机端,同步测试HTML5网页.测试发布软件:FileZilla JavaScript ...

  5. 使用Struts2标签遍历集合

    遍历Map<String,Object> 遍历Map<Stirng,List<Student>> 遍历List<Map<String,Student&g ...

  6. 窗口之间的主从关系与Z-Order

    说明:这是本人2008年写的一篇旧文,从未公开发表过.其中除了一小段描述Window Mobile平台的内容已过时,大部分内容对于从事Win32开发的程序员还是很有参考价值的,也是对自己从事Windo ...

  7. Java经典实例:纪元秒和本地日期时间互换

    Java版本:1.8开始 import java.time.Instant; import java.time.ZoneId; import java.time.ZonedDateTime; /** ...

  8. 决战大数据之三-Apache ZooKeeper Standalone及复制模式安装及测试

    决战大数据之三-Apache ZooKeeper Standalone及复制模式安装及测试 [TOC] Apache ZooKeeper 单机模式安装 创建hadoop用户&赋予sudo权限, ...

  9. Ionic 开发环境搭建

    android sdk环境搭建并非易事,本人经过无数失败,才使用以下方式成功 配置Ionic 开发环境 1.下载JDK并配置Java运行环境 http://www.oracle.com/technet ...

  10. 仿iphone日历插件(beta)

    前言 小伙伴们好,很久不见了.最近工作进入正常期了,所以慢慢的悠闲的时间久没有了,所以不能每天水一篇了. 最近也在听师傅(http://home.cnblogs.com/u/aaronjs/)的教导开 ...