1、HashMap的数据结构(HashMap通过hashcode对其内容进行高速查找,是无序的)

数据结构中有数组和链表来实现对数据的存储,但这两者基本上是两个极端。

数组 :数组的存储区是连续的,占用内存严重,故空间复杂度非常大。但数组的二分查找时间度小;数组的特点:寻址easy,插入和

删除困难。

链表 :链表的储存区离散。占用内存比較宽松。故空间复杂度非常小,但时间复杂度大;链表的特点:寻址困难,插入和删除easy。

哈希表

HashMap是由数组+链表组成。寻址easy,插入和删除easy。(存储单元数组Entry[],数组里面包括链表)

HashMap事实上也是由一个线性的数组实现的。

所以能够理解为其存储数据的容器就是一个线性容器;

HashMap里面有一个内部静态类Entry,其重要的属性有key,value,next,从属性key,value 就能够非常明显的看出来 Entry就是

HashMap键值对实现的一个基础bean;也就是说HashMap的基础就是一个线性数组,这个数组就是Entry[]。Map里面的内容都保存

在Entry[]中;

/**

* The table, resized as necessary. Length MUST Always be a power of two.

*/

transient Entry[] table;

2、HashMap的存取实现

2.1:存储

这里HashMap用了一个算法。

//存储时候:

int hash=key.hashCode();//获取key的hashCode,这个值是一个固定的int值

int index=hash%Entry[].length。//获取数组下标:key的hash值对Entry数组长度进行取余

Entry[index]=value。

注意:假设两个key通过hash%Entry[].length得到的index同样。会不会覆盖?

是不会的。Entry类有一个next属性,作用是指向下一个Entry。打个例如, 第一个键值对A进来。通过计算其key的hash得到的

index=0。记做:Entry[0] = A。一会后又进来一个键值对B,通过计算其index也等于0,如今怎么办?HashMap会这样做:B.next =

A,Entry[0] = B,假设又进来C,index也等于0,那么C.next = B,Entry[0] = C;这样我们发现index=0的地方事实上存取了A,B,C三个键值对,他

们通过next这个属性链接在一起。

所以疑问不用操心。

也就是说Entry[]数组中存储的是最后插入的数据

public V put(K key, V value) {

if (key == null)

return putForNullKey(value); //null总是放在数组的第一个链表中

int hash = hash(key.hashCode());

int i = indexFor(hash, table.length);

//遍历链表

for (Entry e = table[i]; e != null; e = e.next) {

Object k;

//假设key在链表中已存在,则替换为新value

if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {

V oldValue = e.value;

e.value = value;

e.recordAccess(this);

return oldValue;

}

}

modCount++;

addEntry(hash, key, value, i);

return null;

}

void addEntry(int hash, K key, V value, int bucketIndex) {

Entry e = table[bucketIndex];

table[bucketIndex] = new Entry(hash, key, value, e); //參数e, 是Entry.next

//假设size超过threshold,则扩充table大小。再散列

if (size++ >= threshold)

resize(2 * table.length);

}

2.2:取值

获取key的hashcode指,通过hash值去hash%Entry[].length  获取Entry[hash%Entry[].length],定位到该数组元素之后,再遍历该元

素处的链表。

//取值时候:

int hash=key.hashCode();

int index =hash%Entry[].length;

return Entry[index];

public V get(Object key) {

if (key == null)

return getForNullKey();

int hash = hash(key.hashCode());

//先定位到数组元素。再遍历该元素处的链表

for (Entry e = table[indexFor(hash, table.length)];

e != null;

e = e.next) {

Object k;

if (e.hash == hash && ((k = e.key) == key || key.equals(k)))

return e.value;

}

return null;

}

当哈希表的容量超过默认容量时,必需要调整table的大小。

当容量达到最大值时,该方法Integer.MAX_VALUE返回。这时。就需要创建

一张表,将原来的表映射到新表中。

3、HashMap、HashTable和ConcurrentHashMap的线程安全问题

HashMap:线程不安全的。

HashTable:锁住整张hash表,让线程独占。hashMap同意为空。

通过分析Hashtable就知道,synchronized是针对整张Hash表的,即每次锁住整张表

让线程独占。安全的背后是巨大的浪费。

ConcurrentHashMap:一个更快的hashmap,它提供了好得多的并发性。多个读操作差点儿总能够并发地运行。

他是锁段(默认:把hash表分为16个

段),在get,put,remove等操作中,ConcurrentHashMap仅仅锁定当前须要用到的段,仅仅有在求size的时候才锁定整张hash表。

hashmap储存有向图_HashMap的存储结构及原理相关推荐

  1. hashmap 存取原理图_HashMap的存储原理

    HashMap提供高效的查找,插入和删除.是怎么做到的? HashMap的存储结构 HashMap底层是以数组方式进行存储的.将key-value键值对作为数组的一个元素进行存储. Key-value ...

  2. 机械硬盘的存储结构及原理

    硬盘是电脑主要的存储媒介之一.根据硬盘的读写方式和存储方式不同,当前主流的硬盘可以分为固态硬盘(SSD硬盘).机械硬盘(HDD 硬盘)两种.由于固态硬盘存在价格昂贵.容量较小和一旦损坏难以修复等特点, ...

  3. hashmap储存有向图_java-对后边缘进行运算以获取有向图中的循环数

    我一直在编写代码,以在有向图中获得所有可能的周期. Here是一种跟踪后沿的实现,只要找到一个后沿,它就会返回true,即检测到一个循环.我将其扩展到以下内容: 计算一棵树中所有可能的后边缘,后边缘的 ...

  4. hashmap储存有向图_如何在Rust中构建向量的HashMap?

    将[]放到HashMap上是(现已弃用)get(..)函数的糖,该声明是: fn get(&'a self, k: &K) -> &'a V 并返回一个常量(&) ...

  5. hashmap储存有向图_以邻接表的形式创建带权值的有向图即有向网

    1.有向网的数据结构 单链表中的每个结点有3个域组成,分别为邻接点域(adjvex)表示与某顶点邻接的点在图中的位置:链域(nextarc)表示下一条边或弧的结点:数据域info存储和边或弧相关的信息 ...

  6. 数据结构图之一(基本概念,存储结构,两种遍历)

    [1]图的基本概念 (1)图是由顶点集合以及顶点间的关系集合组成的一种数据结构. Graph = (V,E)  V是顶点的又穷非空集合:E是顶点之间关系的有穷集合,也叫边集合. (2)有向图:顶点对& ...

  7. 图的常见存储结构及各自的优缺点

    以下说法均建立在简单图上,即无环无重复边的图. 本文将介绍图的常见存储结构及各自的优缺点 邻接矩阵 邻接表 十字链表 邻接多重表 边集数组 邻接矩阵 用两个数组来表示图:一个一维数组存储图中顶点信息, ...

  8. Java HashMap原理及内部存储结构

    本文将通过如下简单的代码来分析HashMap的内部数据结构的变化过程. public static void main(String[] args) {Map<String, String> ...

  9. 一个含n个顶点和e条弧的有向图以邻接矩阵表示法为存储结构,则计算该有向图中某个顶点出度的时间复杂度为

    一个含n个顶点和e条弧的有向图以邻接矩阵表示法为存储结构,则计算该有向图中某个顶点出度的时间复杂度为(       ) A.O(n)                                   ...

最新文章

  1. c 语言常用宏定义 模板
  2. C/C++中“空语句”的说明
  3. python中通过pip安装套件
  4. for循环用i++和++i哪个效率高?
  5. Vivado入门使用指南之----多路分频器(逻辑分析仪IP的使用以及前后仿真及ip的基本使用)
  6. [转载] JAVA出现空指针异常(初学者)
  7. 7. 吴恩达机器学习课程-作业7-Kmeans and PCA
  8. (转)策略回测的框架、实现、测试
  9. 重磅预告!企业上云的正确姿势
  10. k2pbreed刷高恪教程_斐讯K2刷高恪固件教程,通过breed刷入,详细图文教程
  11. iOS中 语音识别功能/语音转文字教程详解
  12. matlab半导体器件仿真,半导体软件 - 仿真模拟半导体器件的物理场
  13. wpsmac历史版本_WPS Office 2019 1.2.0(1574) Mac中文正式版
  14. c语言数字大小32768,c语言中int(2byte)的表示范围是为什么是-32768~32767!
  15. org.jboss.netty.util.internal.jzlib.ZStream scanned from multiple locations: jar:
  16. 流利阅读 2019.2.22 Duke University apologizes over professor’s email asking Chinese students to speak En
  17. 低学历者已经月入万元是为什么
  18. Python | 图片转文字
  19. 基于 GitLab CI 的前端工程CI/CD实践
  20. 如何把华为数据分析项目写进简历

热门文章

  1. MOQL—转换器(Translator)
  2. 【廖雪峰官方网站/Java教程】多线程(2)
  3. 从tensorflow的summary中提取数据,并进行平滑操作与显示
  4. Spring整合Struts2的两种方式
  5. LightOJ1245 Harmonic Number (II) —— 规律
  6. 一步一步使用阿里云容器服务部署基于.NET的JEXUS网站 (转)
  7. 看看junit在一个具体的项目中
  8. zip压缩/tar打包
  9. chkconfig、mount、shutdown
  10. Tomcat 8 中的startup.bat