HashMap学习笔记

概述

HashMap是Map接口的一个哈希表的实现,内部是一个数组表示的。数组中的元素叫做一个Node,一个Node可以一个是一个简单的表示键值对的二元组,也可以是一个复杂的TreeNode。如果是一个一个简单的二元组,则可以通过Node的next域构成构成一个链表。

  transient Node<K,V>[] table;
    

当需要遍历Map的时候,建议使用entrySet域来进行遍历,而不是keySet。因为entrySet其实返回的是一个特殊的set,这个set并未保存任何元素,而是定义了一个特殊的迭代器,这个迭代器直接遍历map中的table,因此能直接得到一个entry的键和值,也就是说,使用entrySet进行遍历的时候,一次遍历即可。(hash冲突除外)。而使用keyset得到的只是key的set,使用key去获取value的时候,还需要进行一次查找。

如果只需要值,那么就调用values方法,此方法返回一个特殊的集合,它保存了一个entryset,并使用entryset的迭代器对table进行访问,因此效率也是很高的。

方法详述

  • comparableClassFor
    Class<?> comparableClassFor(Object x)

    获取对象x的可比较的class,即尝试从对象x中找到一个Comparable<x.getClass()> 对象。

    首先检查对象x是否是一个comparable的实例:

    if (x instanceof Comparable)

    如果是的话,获取x的class,顺便检查一下是否是String,因为Stirng是用的最多的,顺便加个速。

    如果不是String类,那么就获取此类直接实现的接口(注意,一定是直接实现的)。

    c.getGenericInterfaces())

    然后判断每个父接口是否是参数化类型,且参数化类型的具体类型为compare:

    if (((t = ts[i]) instanceof ParameterizedType) &&

    ((p = (ParameterizedType)t).getRawType() ==

    Comparable.class)

    如果是的话,取出此参数化类型的实际参数类型,如果确实存在,且参数类型和对象x的类型相同,那么这个父接口就是最终要找的comparable接口了。

    (as = p.getActualTypeArguments()) != null &&

    as.length == 1 && as[0] == c) //

  • tableSizeFor

          int tableSizeFor(int cap) int n = cap - 1; n |= n >>> 1; n |= n
            >>> 2; n |= n >>> 4; n |= n >>> 8; n |= n >>>
            16; return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY
            : n + 1;
            
        

    这段代码非常巧妙,通过位移和或运算计算出比cap大的最小的2的N次方的值。
    第一行:cap-1是为了解决cap本来就是2的N次方的问题。

    然后进行一次右移,一次右移的效果是:如果某一为本来是1,那么一次位移并或之后,它的低一位也会变成1.这样就将一个1变为2个1了。如果本来是0,那么还是0.

    第一步已经将一个1变为2个1了,那么第二步右移两位,就能将两个1变为4个1,接着变为8个1,16个1,最多不超过32个1.

    这个数字最终的效果是,从低往高看,后面的位数上全是1,然后这个数加1,就是原始的cap的2的N次方最小值了。

  • putVal

    第一步:找到带加入的key所在的桶,使用的办法是用桶数组的大小减1然后和hey的哈希值进行与,这其实是让key的哈希值对桶的大小取余的简便操作,提升了效率。

    第二步:如果桶中当前没有元素,则加入,完成。如果桶中已经有值了,说明发生了hash冲突,此时需要将新加入的值沿着桶中已有的值往下排,构成一个链表。

    在第二中,如果一个桶中的元素超过了一个阀值,默认为8,则不再使用链表来保存桶中的元素,而是改成用二叉查找树来保存,以提高检索效率。

    如果桶中的元素当前已经是一个二叉查找树了,那么就加新元素加入到树中。

    加入一个新元素之后,需要检查Map的数量是否满足填装因子的限制,如果不满足了,需要进行重新hash。

时间: 2024-11-01 01:17:38

HashMap学习笔记的相关文章

java/android 设计模式学习笔记(一)---单例模式

前段时间公司一些同事在讨论单例模式(我是最渣的一个,都插不上嘴 T__T ),这个模式使用的频率很高,也可能是很多人最熟悉的设计模式,当然单例模式也算是最简单的设计模式之一吧,简单归简单,但是在实际使用的时候也会有一些坑. PS:对技术感兴趣的同鞋加群544645972一起交流 设计模式总目录 java/android 设计模式学习笔记目录 特点 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 单例模式的使用很广泛,比如:线程池(threadpool).缓存(cache).对

storm学习笔记完整记录(一)

storm有两种运行模式(本地模式和集群模式) 1. 首先创建一个类似于HelloWorld的简单程序,以便进入storm的大门,包结构如下: 2.从包结构可以知道,这是一个Maven Project,pom.xml的内容如下: <project xmlns="http://maven.apache.org/POM/4.0.0"          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"    

Java集合框架学习笔记

本文为学习笔记,学习课程为慕课网Java入门第三季中的集合框架部分,若需要研究文中的代码,可前往下载.http://www.imooc.com/learn/110 1. List(Collection子接口) 1.1 实例化 List list = new ArrayList(); ??List是一个接口,不可直接实例化,通常情况下ArrayList实现类进行实例化. 1.2 增 1.2.1 add(obj) ??直接将obj对象加入List末位. 1.2.2 add(i, obj) ??将ob

【DAY12】第十二天集合&泛型&IO学习笔记

hash:散列 ------------------ Hashset集合内部是通过HashMap进行实现的.使用的是HashMap中key部分. 对象在添加进集合中时,首选会对hashcode进行处理(hashcode右移16位和 自身做异或运算)得到一个经过处理的hash值,然后该值和集合的容量进行 &运算,得到介于0和集合容量值之间一个数字.该数字表示着数组的下标. 也就是该元素应该存放在哪个元素中. Map与Collection -------------- Map与Collection在

JPA学习笔记(5)——EntityManager相关

Persistence EntityManagerFactory EntityManager find方法 getReference方法 persist方法 remove方法 merge方法 情况1传入的对象没有id 情况2传入的对象有identityManager的缓存中没有该对象数据库中没有该记录 情况3传入的对象有identityManager的缓存没有该对象数据库中有该记录 情况4传入的对象有identityManager的缓存有该对象 flush方法 refresh方法 clear c

SpringMVC学习笔记(二): 日常使用功能

前提: 1.web.xml 和spring-mvc核心配置如:SpringMVC学习笔记(一): 基础知识中注解实现. 2.类的@RequestMapping(value="/annotationController") 3.spring-mvc 推荐使用注解实现. 一.数据的接收 (一)URL参数数据的接收 1.使用 HttpServletRequest 获取参数 <span style="font-size:18px;"><span style

java学习笔记10--泛型总结

java学习笔记系列: java学习笔记9--内部类总结 java学习笔记8--接口总结 java学习笔记7--抽象类与抽象方法 java学习笔记6--类的继承.Object类 java学习笔记5--类的方法 java学习笔记4--对象的初始化与回收 java学习笔记3--类与对象的基础 java学习笔记2--数据类型.数组 java学习笔记1--开发环境平台总结 本文地址:http://www.cnblogs.com/archimedes/p/java-study-note10.html,转载

不错的Spring学习笔记(转)

Spring学习笔记(1)----简单的实例 ---------------------------------   首先需要准备Spring包,可从官方网站上下载.   下载解压后,必须的两个包是spring.jar和commons-logging.jar.此外为了便于测试加入了JUnit包.   在Myeclipse中创建Java项目.   编写一个接口类,为了简单,只加入了一个方法.   Java代码   1.package com.szy.spring.interfacebean;  

黑马程序员_JAVA 基础加强学习笔记

一.面向对象 (一)继承  1.继承的好处: (1) 提高了代码的复用性. (2) 让类与类之间产生了关系,提供了另一个特征多态的前提. 注意: 子类中所有的构造函数都会默认访问父类中的空参数的构造函数,因为每一个子类构造内第一行都有默认的语句super();  如果父类中没有空参数的构造函数,那么子类的构造函数内,必须通过super语句指定要访问的父类中的构造函数. 如果子类构造函数中用this来指定调用子类自己的构造函数,那么被调用的构造函数也一样会访问父类中的构造函数. 2.final特点