Map,百度翻译给我的解释是映射,在Java编程中,它是存储键值对(key-value)的一种容器,也是Java程序员常用的对象。这篇博客介绍下HashMap的实现;java是面向对象编程语言,jdk为java提供了丰富的API,自然而然,在java中,数据的展示形式也是多种多样的。但是在底层语言,数据的展示就不同了,一般只有两种形式,元素值(基本类型)、数组,其他的数据类型都是这两个元素的封装,今天要说的HashMap就是如此。

  HashMap通过数组对数据进行存储的(Entry[] table)。其中,数组元素对象Entry是HashMap中的一个内部类,这个内部类记录了两个关键的东西,一个是键值对;一个next,下一个Entry。下面我介绍下将一个键值对put到HashMap中的过程;将key,value put到HashMap中,

1、拿到当前这个key的hashcode,对hashcode进行空间稀疏处理,最后得到的还是一个码,这个码在同一个字节位置的不同的整数倍上是不同的;

2、拿到上一步获得的hashcode,与当前表的长度进行与计算(h & (length-1)),获取当前key在当前数组中的坐标;

3、拿到上一步中与运算获得的坐标,在表中以阈值为单位长度,相同的位置进行查找,寻找是否存在相同的key,如果存在相同的key,那么就将key进行值替换,如果没有找到key,那么进行添加;

  for (Entry<K,V> e = table[i]; e != null; e = e.next) {
Object k;
if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
V oldValue = e.value;
e.value = value;
e.recordAccess(this);
return oldValue;
}
}

在这里说一下HashMap是如何进行相同的Key的查找的。在jdk中,给对象都赋予了一个身份标识,hashcode,在java中,因为一个对象在不同的特征形态下是允许有不同的hashcode的,所以,进行了第一步处理,使得一个对象只有一个hashcode;而Hashmap存储数据是通过数组,所以找制定的键值对,实际上就是找我要的key所在的键值对的位置;为此,hashmap在存储键值对时候,存储位置是根据key的hashcode和当前hashmap内存储数组的长度来进行转换而得到的(position = hashcode & (length - 1))。举个例子,实例化一个HashMap的时候,会声明存储数组table的初始化大小为16;而我要存储一个数键值对{0:"0"},那么我存储的这个键值对的位置就 0.hashcode() & (16 -1) =0;这个键值对就是存储在Hashmap中的table数组中的第一个元素;现在我再想hashmap中put进去一个元素{5:“5”},此时5.hashcode() & (16 - 1)=5,那么,我put的key为5的这个键值对存储在hashmap中的数组table中的位置就是5;然后我在想hashmap中put进去一个元素{16:16},那么我再去数组中,发现得到位置是0,但是当前这个数组,0这个位置上已经存在数了,而且还没有找到我要存的key,那么,再往这个对象中塞键值对,就用到上面我已经介绍过的Entry对象的属性,next了。

 /**
* Adds a new entry with the specified key, value and hash code to
* the specified bucket. It is the responsibility of this
* method to resize the table if appropriate.
*
* Subclass overrides this to alter the behavior of put method.
*/
void addEntry(int hash, K key, V value, int bucketIndex) {
Entry<K,V> e = table[bucketIndex];
table[bucketIndex] = new Entry<>(hash, key, value, e);
if (size++ >= threshold)
resize(2 * table.length);
}

  上面的添加的代码,就是如果table的当前这个位置已经有元素了,这时候,你还再往这个位置上添加新元素时,这个新元素就会放到新的元素放在数组中,把之前这个位置已经存在的元素放在当前这个新元素的next上,如果后面再有新元素新增进来,这次的元素就会到后台,作为更新元素的next,那个更新的元素存储到数组table中,这样在第一段代码找可以的时候循环时,就不用挨个遍历数组元素了,只需要遍历阈值相关的制定倍数的数据,效率就高了很多,后面的get方法,用到的额也是这里put元素时的找数的方法。

这篇博客主要就是介绍Hashmap的算法,其他的细节的实现建议去看JDK源码;

HashMap 取数算法的更多相关文章

  1. codevs 1907 方格取数 3

    Description 在一个有m*n 个方格的棋盘中,每个方格中有一个正整数.现要从方格中取数,使任意2 个数所在方格没有公共边,且取出的数的总和最大.试设计一个满足要求的取数算法. Input 第 ...

  2. 【wikioi】1907 方格取数3(最大流+最大权闭合子图)

    http://www.wikioi.com/problem/1907/ 这题我一开始想到的是状压,看到n<=30果断放弃. 然后也想到了黑白染色,然后脑残了,没想到怎么连边. 很简单的一题 黑白 ...

  3. 【网络流24题】No.9 方格取数问题 (二分图点权最大独立集)

    [题意] 在一个有 m*n 个方格的棋盘中, 每个方格中有一个正整数. 现要从方格中取数, 使任意 2 个数所在方格没有公共边,且取出的数的总和最大.试设计一个满足要求的取数算法. 输入文件示例inp ...

  4. 洛谷 P2774 方格取数问题 解题报告

    P2774 方格取数问题 题目背景 none! 题目描述 在一个有 \(m*n\) 个方格的棋盘中,每个方格中有一个正整数.现要从方格中取数,使任意 2 个数所在方格没有公共边,且取出的数的总和最大. ...

  5. LibreOJ #6007. 「网络流 24 题」方格取数 最小割 最大点权独立集 最大流

    #6007. 「网络流 24 题」方格取数 内存限制:256 MiB时间限制:1000 ms标准输入输出 题目类型:传统评测方式:文本比较 上传者: 匿名 提交提交记录统计讨论测试数据   题目描述 ...

  6. 【刷题】LOJ 6007 「网络流 24 题」方格取数

    题目描述 在一个有 \(m \times n\) 个方格的棋盘中,每个方格中有一个正整数. 现要从方格中取数,使任意 \(2\) 个数所在方格没有公共边,且取出的数的总和最大.试设计一个满足要求的取数 ...

  7. Libre 6007 「网络流 24 题」方格取数 / Luogu 2774 方格取数问题 (网络流,最大流)

    Libre 6007 「网络流 24 题」方格取数 / Luogu 2774 方格取数问题 (网络流,最大流) Description 在一个有 m*n 个方格的棋盘中,每个方格中有一个正整数.现要从 ...

  8. P2774 方格取数问题

    题目背景 none! 题目描述 在一个有 m*n 个方格的棋盘中,每个方格中有一个正整数.现要从方格中取数,使任意 2 个数所在方格没有公共边,且取出的数的总和最大.试设计一个满足要求的取数算法.对于 ...

  9. codevs1907 方格取数 3

    «问题描述:在一个有m*n 个方格的棋盘中,每个方格中有一个正整数.现要从方格中取数,使任意2 个数所在方格没有公共边,且取出的数的总和最大.试设计一个满足要求的取数算法.«编程任务:对于给定的方格棋 ...

随机推荐

  1. codeforces 161 D. Distance in Tree(树形dp)

    题目链接:http://codeforces.com/problemset/problem/161/D 题意:给出一个树,问树上点到点的距离为k的一共有几个. 一道简单的树形dp,算是一个基础题. 设 ...

  2. bzoj5072 小A的树 题解

    题意 给出一棵 n 个点的树,每个点有黑白两种颜色.q 次询问,每次 询问给出 x 和 y,问能否选出一个 x 个点的联通子图,使得其中 黑点数目为 y. 范围 n ≤ 5000,q ≤ 10^5 其 ...

  3. 洛谷 P1980【计数问题】 题解(2)

    还有一种办法,就是用stringstream函数将每一次的数全都转化成char一维数组样式的字符串,然后逐位扫一遍即可. (记得判断字符时将规定数字+48) //Stand up for the fa ...

  4. explain的关键字段的意义

    mysql提供的explain工具可以输出一些有用的信息. 一下是相关的部分返回值的意义. select_type 表示SELECT的类型,常见的取值有: SIMPLE:简单表,不使用表连接或子查询 ...

  5. Windows7关闭默认共享

    注意:禁用默认共享会导致安装ORACLE时执行先决条件检测不通过,要想安装ORACLE得先打开共享. 1.win+r 运行里输入compmgmt.msc进入计算机管理,在共享文件夹里,去掉所有的盘共享 ...

  6. 07 (OC)* XIB原理和Xib、storyBoard、代码的优缺点

    1:可读性 2:可视化界面.立马看到 3:开发速度. 4:复用性 5:维护性差 本质 编译时对xml文件做了如下操作1,读取xml文件,生成所有界面对象,生成所有object(即自定义的control ...

  7. 利用IntelliJ IDEA与Maven开发scala程序,并打包提交到spark集群

    https://zhuanlan.zhihu.com/p/23141509 https://blog.csdn.net/u011470552/article/details/54564636 http ...

  8. 细谈Redis五大数据类型

    文章原创于公众号:程序猿周先森.本平台不定时更新,喜欢我的文章,欢迎关注我的微信公众号. 上一篇文章有提到,Redis中使用最频繁的有5种数据类型:String.List.Hash.Set.SortS ...

  9. github博客Hexo引流到微信

    相信有不少小伙伴都在github上创建了属于自己的博客,其中用Hexo的Next主题应该不少,那么,我们究竟该如何将博客的流量引流到微信呢?今天就来带你看一看. 如何引流 现在网上有一种套路,当你在看 ...

  10. 基于RDD实现简单的WordCount程序

    写在前面 因为觉得自己的代码量实在是太少了,所以,想着,每周至少写5个小的demo程序.现在的想法是,写一些Spark,Storm,MapReduce,Flume,kafka等等单独或组合使用的一些小 ...