1、为什么需要HashMap

前面我们说了ArrayList和LinkedList,它们对容器内的对象都能实现增、删、改、查、遍历等操作,

并且对应不同的情况,我们可以选择不同的List,用以提高效率。从功能上来说,这个容器已经设计得很好了,

为什么我们还要HashMap呢?接下来,让我们细细地分析:

在程序中,我们最常用的动作就是存数据和取数据,存数据就是把数据存起来(包括插入数据和增加数据),

取数据就是找到我们存进去的数据(查找数据),下面从这两个方面来分析一下ArrayList和LinkedList的问题。

  • ArrayList在增加数据和按下标获取数据效率高。
  • LinkedList在插入数据和删除数据方面效率比较高。

提出问题:能不能有一种容器能在它存数据的时候很快,查找数据的时候也很快呢?

新的需求:事物之间都是有关联的。ArrayList是靠下标来关联数据的,这种关联太单一了,

无法完美的适应OO编程环境,面向对象编程就是要把世间万物都用对象包装起来,

我们把这些对象都只与数字(0、1、2)关联起来,也太单调了,也太没有颜色了。

世间是多彩的,基于这种情况,我们就提出:

能不能使其他的事物也与我们的对象关联起来呢?(比如说,我们用汽车牌号码关联汽车,用身份证号码关联人)

当然同时满足上面提出的条件(能不能有一种容器能在它存数据的时候很快,查找数据的时候也很快呢?),

那就非常完美了。

程序袁的回答:是的,我们能过设计出这种容器,他将马上风靡全球。

2、HashMap实现原理

先看看下面这两张图

表1这是HashMap的两种存储模式表二

下面解释这两种表存储对象的原理

首先理解两个概念:

  • hashcode(散列码):现在只需要知道hashcode是返回一个整数。

后面会单独用一章具体解析hashcoed和equal的问题,以及怎么写hashcode和equal的。

  • equal(相等):现在只需要知道equal在两个逻辑上面相等的对象的时候返回true,

例如:new student(1,"aa");和new student(1,"aa")是两个相等的对象,如果用object里面的equal来判断

这两个对象结果是不相等的。

存储数据的分析:

现在有以下几个对象:

a=newnew student(1,"aa"); b=new
student(4,"aa");

c=new
student(2,"aa");d=new student(1,"aa")

现在假设a对象的hashcode是5,b的hashcode是17,假设 c的hashcode是10,,

因为一些规则的原因(在hashcode这一章会重点解析),d的hashcode必须是5;接下来我们就对

这些数据进行存储。先对表一进行存储:int index=hashcode()%12--------这个12是值这个表的大小。

先存a,那么a对象放在5这个位置,接下来存b,此时经过int
index=hashcode()%12,

也会存储到5这个位置,但是此时此刻,5这个位置已经有数据了,,接下来判断这个a对象

跟b对象是否相等,现在是不想等,那我们就把这个b对象放到右边这个非常小的格子里面的第一个格子,

接下来存储c对象,把c放在10这个地方,再接下来存储d,经过计算吗,d会存储到5这个位置,

但是此时此刻,5这个位置已经有数据了,接下来判断这个a对象跟b对象是否相等,现在是相等,

那我们就把d这个对象替换掉原来a的对象。

存储结果如下图:

注意5这个位置的元素的替换

接下来对表二的存储方式进行分析:公式也是一样的int index=hashcode()%12,先存a,那么a对象放在5这个位置,

接下来存b,此时经过int index=hashcode()%12,也会存储到5这个位置,但是此时此刻,

5这个位置已经有数据了,接下来判断这个a对象跟b对象是否相等,现在是不想等,

那我们就把这个b对象放在5这个位置接在a的对象后面,接下来存储c对象,把c放在10这个地方,

再接下来存储d,经过计算吗,d会存储到5这个位置,但是此时此刻,5这个位置已经有数据了,

接下来是遍历5这一条链路的数据,看看是否有相等的对象,现在找到a这个对象是相等的,

那我们就把d这个对象替换掉原来a的对象。存储结果如下图。

注意5这个位置a和d的替换

查找数据的分析:

以上是存储对象时的分析,接下来是查找对象的分析,(以下分析是针对表一进行分析的)现在我要查找三个对象,

x=newnew
student(1,"aa"), y=newnew student(3,"aa"),z=new
student(4,"aa"),

由相等的对象必须有相同的hashcode这个条件的限制(先不管为什么有这个霸道的条件,后面会解析),

我们可知x的hashcode的为5,z的hashcode为17,y的hashcode我们就随便取一个,假设为10,

(为什么不同的对象会有相同的hashcode?这样行吗?合理吗?后面会解析)。接下来我们就来查找对象了。

先查找x,经过int
index=hashcode()%12计算,是第5格,此时有数据,接下来判断这两个对象是否相等,

此时相等,那么我们就找到了这个对象。接下来查找y,经过int
index=hashcode()%12计算是第10个,

此时有数据,接下来判断这两个对象是否相等,此时不相等,然后就到右边那个

非常小的格子里面一个一个的判断是否相等,我们遍历一遍后,发现没有相等的,此时我们得出结论,没有这个数据,

再接下来找z,经过int
index=hashcode()%12计算是第10个,此时有数据,接下来判断这两个对象是否相等,此时不相等,

然后就到右边那个非常小的格子里面一个一个的判断是否相等,在遍历的时候,

我们发现第一个就是相等的,那我们就找到了这个数据。

3、总结HashMap的主要特点

    • 如果我们不知道下标,相比ArrayList和LinkedList,hashMap有具有非常快的查找能力
    • hashMap不仅能够形成键值对,这个是一个非常好的关系

java容器HashMap原理的更多相关文章

  1. 理解java容器底层原理--手动实现HashMap

    HashMap结构 HashMap的底层是数组+链表,百度百科找了张图: 先写个链表节点的类 package com.xzlf.collection2; public class Node { int ...

  2. Java基础-hashMap原理剖析

    Java基础-hashMap原理剖析 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.   一.什么是哈希(Hash) 答:Hash就是散列,即把对象打散.举个例子,有100000条数 ...

  3. java中HashMap原理?

    参考:https://www.cnblogs.com/yuanblog/p/4441017.html(推荐) https://blog.csdn.net/a745233700/article/deta ...

  4. Java学习----HashMap原理

    1.HashMap的数据结构 数组的特点是:寻址容易,插入和删除困难:而链表的特点是:寻址困难,插入和删除容易.那么我们能不能综合两者的特性,做出一种寻址容易,插入删除也容易的数据结构?答案是肯定的, ...

  5. 容器HashMap原理(学习)

    一.概述 基于哈希表的 Map 接口的非同步实现,允许使用 null 值和 null 键,不保证映射的顺序 二.数据结构 HashMap实际上是一个“链表散列”的数据结构,即数组和链表的结合体:Has ...

  6. 理解java容器底层原理--手动实现HashSet

    HashSet的底层其实就是HashMap,换句话说HashSet就是简化版的HashMap. 直接上代码: package com.xzlf.collection2; import java.uti ...

  7. 深入理解Java容器——HashMap

    目录 存储结构 初始化 put resize 树化 get 为什么equals和hashCode要同时重写? 为何HashMap的数组长度一定是2的次幂? 线程安全 参考 存储结构 JDK1.8前是数 ...

  8. java的HashMap 原理

    https://www.cnblogs.com/chengxiao/p/6059914.html 哈希表(hash table)也叫散列表,是一种非常重要的数据结构,应用场景及其丰富,许多缓存技术(比 ...

  9. 理解java容器底层原理--手动实现LinkedList

    Node java 中的 LIinkedList 的数据结构是链表,而链表中每一个元素是节点. 我们先定义一下节点: package com.xzlf.collection; public class ...

随机推荐

  1. Java编程的逻辑 (55) - 容器类总结

    本系列文章经补充和完善,已修订整理成书<Java编程的逻辑>,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营链接:http:/ ...

  2. Codeforces Round #475 (Div. 2) D. Destruction of a Tree

    题意:给你一棵树, 只能删度数为偶数的点, 问你能不能将整个图删完, 如果能输入删除的顺序. 思路:对于一棵树来说, 如果里面的点的个数是偶数个则肯定不可能, 偶数个点有奇数条边,而你每次删只能删偶数 ...

  3. zookeeper命令行客户端

    前提条件:搭建好zookeeper服务器集群<Zookeeper深入认识>,并且集群成功开启. 执行zkServer.sh,客户端连接上服务器hadoop1. 都有哪些命令行操作呢?(见下 ...

  4. 【Ray Tracing in One Weekend 超详解】 光线追踪1-3

    学完了插值,我们来学习在场景里面添加一个立体彩色球(三维插值) 按照惯例,先看效果: Chapter4: Adding a sphere 我们又一次面临图形学的主要任务. 我们需要再次回顾coord1 ...

  5. Android 常见SD卡操作

    目录 Android 常见SD卡操作 Android 常见SD卡操作 参考 https://blog.csdn.net/mad1989/article/details/37568667. [0.] E ...

  6. lupgu P3950 部落冲突

    题目链接 luogu P3950 部落冲突 题解 树剖线段树可以 lct还行 代码 #include<cstdio> #include<algorithm> inline in ...

  7. HDU.5181.numbers(DP)

    题目链接 参考. \(Description\) 将\(1,2,\cdots,n(n\leq 300)\)依次入栈/出栈,并满足\(m(m\leq 90000)\)个形如\(x\)要在\(y\)之前出 ...

  8. HDU.5985.Lucky Coins(概率DP)

    题目链接 \(Description\) 有n(n<=10)种硬币,已知每种硬币的数量和它抛一次正面朝上的概率pi.进行如下过程:每次抛一次所有硬币,将正面朝下的硬币去掉.重复该过程直到只剩一种 ...

  9. 如何处理C++构造函数中的错误——兼谈不同语言的错误处理

    用C++写代码的时候总是避免不了处理错误,一般来说有两种方式,通过函数的返回值或者抛出异常.C语言的错误处理一律是通过函数的返回值来判断的,一般是返回0.NULL或者-1表示错误,或者直接返回错误代码 ...

  10. Code Forces 698A Vacations

    题目描述 Vasya has nn days of vacations! So he decided to improve his IT skills and do sport. Vasya know ...