HashMap源代码解析

  HashMap原理剖析

  之前有看过别人的HashMap源代码的分析,今天尝试自己来分析一波,纯属个人愚见。听一些老的程序员说过,当别人跟你说用某样技术到项目中去,而你按照别人的想法实现了的时候,你只能是一个码农,当你自己会想到用一样东西到你的实际开发中的时候,你是一个普通的程序员,当你不仅能想到用某样技术到项目中去,而且深深的熟悉这项技术的底层实现,那就是一个有内功的程序员了。

  HashMap,哈希表,基于哈希算法实现的一个java集合。起初使用这个哈希表的时候,感觉非常的不习惯,主要是容易和其他的几种哈希表搞混,就当时的理解来说,哪个哈希表的键不能为空,哪个哈希表的值可以为空等等,觉得这些东西记住了就感觉自己非常的牛逼了。但是随着学习的进一步深入,总会有疑问,哈希表到底是如何实现的?那就按照api的方法来一个一个解释。

构造方法:

1.空参数的构造方法

 1      public HashMap() {
 2             //初始化加载因子,等于默认的加载因子0.75
 3             this.loadFactor = DEFAULT_LOAD_FACTOR;
 4             //最大容量为初始化容量乘以加载因子
 5             threshold = (int)(DEFAULT_INITIAL_CAPACITY * DEFAULT_LOAD_FACTOR);
 6             //初始化一个长度为16的entry数组,也就是存放键值对的数组
 7             table = new Entry[DEFAULT_INITIAL_CAPACITY];
 8             //调用init方法
 9             init();
10         }

2.初始化容量的构造方法

1 public HashMap(int initialCapacity) {
2        //调用两个参数的构造方法,初始化容量和默认加载因子
3        this(initialCapacity, DEFAULT_LOAD_FACTOR);
4 }

3.初始化容器和初始化加载因子的构造方法

public HashMap(int initialCapacity, float loadFactor) {
            if (initialCapacity < 0)
                throw new IllegalArgumentException("Illegal initial capacity: " +
                                                   initialCapacity);
            if (initialCapacity > MAXIMUM_CAPACITY)
                initialCapacity = MAXIMUM_CAPACITY;
            if (loadFactor <= 0 || Float.isNaN(loadFactor))
                throw new IllegalArgumentException("Illegal load factor: " +
                                                   loadFactor);

            int capacity = 1;
            while (capacity < initialCapacity)
                capacity <<= 1;

            this.loadFactor = loadFactor;
            threshold = (int)(capacity * loadFactor);
            table = new Entry[capacity];
            init();
        }

4.Map集合作为参数的构造方法

1 public HashMap(Map<? extends K, ? extends V> m) {
2     //调用初始化加载因子和初始化容量两个参数的构造方法
3     this(Math.max((int) (m.size() / DEFAULT_LOAD_FACTOR) + 1,
4                DEFAULT_INITIAL_CAPACITY), DEFAULT_LOAD_FACTOR);
5     //将参数Map中的key-value对放入新创建的map集合中
6     putAllForCreate(m);
7}

存放键值对的方法:

 1 public V put(K key, V value) {
 2         //如果key为空,则调用放入null key的方法,占用entry数组的一个位置
 3         if (key == null)
 4             return putForNullKey(value);
 5         //根据key值计算出hash码值
 6         int hash = hash(key.hashCode());
 7         //并根据哈希码的值计算出该key-value在entry数组中的存放位置
 8         int i = indexFor(hash, table.length);
 9         //根据索引直接定位到那个元素并且从那个元素开始遍历
10         for (Entry<K,V> e = table[i]; e != null; e = e.next) {
11             Object k;
12             if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
13                 V oldValue = e.value;
14                 e.value = value;
15                 e.recordAccess(this);
16                 return oldValue;
17             }
18         }
19
20         modCount++;//修改次数加1
21         addEntry(hash, key, value, i);//将key-value对添加进入entry数组
22         return null;
23     }            
时间: 2024-10-05 11:01:46

HashMap源代码解析的相关文章

android7.x Launcher3源代码解析(3)---workspace和allapps载入流程

Launcher系列目录: 一.android7.x Launcher3源代码解析(1)-启动流程 二.android7.x Launcher3源代码解析(2)-框架结构 三.android7.x Launcher3源代码解析(3)-workspace和allapps载入流程 前两篇博客分别对Lancher的启动和Launcher的框架结构进行了一些分析.这一篇.将着重開始分析界面的载入流程. 1.总体流程 先上一张总体的流程图吧.(图片看不清能够下载下来看或者右击新开个页面查看图片) wate

Spring源代码解析

Spring源代码解析(一):IOC容器:http://www.iteye.com/topic/86339 Spring源代码解析(二):IoC容器在Web容器中的启动:http://www.iteye.com/topic/86594 Spring源代码解析(三):Spring JDBC:http://www.iteye.com/topic/87034 Spring源代码解析(四):Spring MVC:http://www.iteye.com/topic/87692 Spring源代码解析(五

Spring源代码解析(收藏)

Spring源代码解析(收藏)Spring源代码解析(一):IOC容器:http://www.iteye.com/topic/86339 Spring源代码解析(二):IoC容器在Web容器中的启动:http://www.iteye.com/topic/86594 Spring源代码解析(三):Spring JDBC:http://www.iteye.com/topic/87034 Spring源代码解析(四):Spring MVC:http://www.iteye.com/topic/8769

NIO框架之MINA源代码解析(二):mina核心引擎

NIO框架之MINA源代码解析(一):背景 MINA的底层还是利用了jdk提供了nio功能,mina仅仅是对nio进行封装.包含MINA用的线程池都是jdk直接提供的. MINA的server端主要有accept.processor.session三部分组成的.当中accept主要负责在指定的port监听.若有新连接则建立一个新的session.processor则负责处理session相应的发送数据和接收数据并调用上层处理:session则缓存当前连接数据. MINA採用了线程懒启动的技术,即

Android源代码解析之(三)--&amp;gt;异步任务AsyncTask

转载请标明出处:一片枫叶的专栏 上一篇文章中我们解说了android中的异步消息机制. 主要解说了Handler对象的使用方式.消息的发送流程等.android的异步消息机制是android中多任务处理的基础,Handler是整个android应用层体系异步消息传递的基础组件,通过对Handler源代码的解析的解析相信大家对android中的异步消息机制有了一个大概的了解.很多其它关于android中的异步消息机制的知识可參考我的:android源代码解析之(二)–>异步消息机制 android

Spark MLlib LDA 源代码解析

1.Spark MLlib LDA源代码解析 http://blog.csdn.net/sunbow0 Spark MLlib LDA 应该算是比較难理解的,当中涉及到大量的概率与统计的相关知识,并且还涉及到了Spark GraphX图计算方面的知识.要想明确当中的原理得要下一番功夫. LDA源代码解析前的基础知识: 1)LDA主题模型的理论知识 參照:LDA数学八卦 2)SparkGraphX 基础知识 http://blog.csdn.net/sunbow0/article/details/

thttpd源代码解析 定时器模块

thttpd源代码解析 定时器模块 thttpd是很轻量级的httpserver,可运行文件仅50kB.名称中的第一个t表示tiny, turbo, 或throttling 与lighttpd.memcached.redis相比很小巧,仅有不到8k行,而后三者大小分别为:60k,13k,86k 支持HTTP/1.1和CGI:採用IO复用实现,单线程,可移植:实现了基于URL的文件流量限制功能 特别适用于大量静态数据訪问的场景,如图片存储 2004年已经停止维护,有一个关于X-Forwarded-

GlusterFS源代码解析 —— GlusterFS 日志

Logging.c: /* Copyright (c) 2008-2012 Red Hat, Inc. <http://www.redhat.com> This file is part of GlusterFS. This file is licensed to you under your choice of the GNU Lesser General Public License, version 3 or any later version (LGPLv3 or later), or

Andfix热修复框架原理及源代码解析-上篇

热补丁介绍及Andfix的使用 Andfix热修复框架原理及源代码解析-上篇 Andfix热修复框架原理及源代码解析-下篇 1.不知道怎样使用的同学,建议看看我上一篇写的介绍热补丁和Andfix的使用,这样你才有一个大概的框架.通过使用Andfix,事实上我们心中会有一个大概的轮廓,它的工作原理,大概就是.所谓的补丁文件.就是通过打包工具apkpatch比对新的apk和旧的apk之间的差异. 然后让我们的旧包执行的时候.就载入它,把曾经的一些信息替换掉. 我们如今就抱着这个慷慨向去深入源代码探个