首页 > 其他分享 >简单剖析Hashmap

简单剖析Hashmap

时间:2024-01-20 20:01:10浏览次数:20  
标签:Node map hash HashMap 剖析 哈希 简单 public Hashmap

剖析 Java Hashmap 源码

在 Java 的集合框架中,HashMap 是一颗璀璨的明珠。通过深入挖掘其源码,我们将揭开 HashMap 的神秘面纱,理解其底层原理、扩容机制和数据结构。

1. HashMap 源码导读

我们首先来看一段简单的代码,创建一个空的 HashMap:

import java.util.HashMap;

public class HashMapSource {

    public static void main(String[] args) {
        // 创建一个空的 HashMap
        HashMap<String, Integer> map = new HashMap<>();
    }
}

2. HashMap 底层原理 - 数组与链表

HashMap 的底层结构是一个数组,每个数组元素是一个链表的头节点。当我们添加键值对时,首先计算键的哈希码,然后通过哈希码找到对应的数组索引。如果发生哈希冲突,即不同键的哈希码映射到同一个索引,就在该索引处形成一个链表。这就是链表法处理碰撞的方式。

下面是一个简化的示例,演示了如何计算哈希码和确定数组索引:

import java.util.HashMap;

public class HashingExample {

    public static void main(String[] args) {
        HashMap<String, Integer> map = new HashMap<>();

        // 计算键 "Alice" 的哈希码
        int hash = "Alice".hashCode();

        // 根据哈希码确定数组索引
        int index = hash & (map.size() - 1);

        System.out.println("键 \"Alice\" 的哈希码: " + hash);
        System.out.println("对应的数组索引: " + index);
    }
}

3. 扩容机制 - Ensuring Capacity

HashMap 在扩容时,会将数组的长度翻倍,并重新计算每个元素的索引。这一过程在 resize() 方法中实现。我们看一下简化版本的代码:

import java.util.HashMap;

public class ResizeExample {

    public static void main(String[] args) {
        HashMap<String, Integer> map = new HashMap<>(16);

        // 添加键值对,触发扩容
        map.put("Alice", 95);
        map.put("Bob", 87);
        map.put("Charlie", 92);
    }
}

在这个例子中,初始容量为 16,当添加第四个键值对时,触发了扩容操作。

4. 数据结构 - Node 类的奥秘

HashMap 的每个键值对都存储在一个 Node 类的实例中。这个类有 hashkeyvaluenext 四个字段,用于保存哈希码、键、值以及下一个节点的引用。以下是一个简化的 Node 类:

class Node<K, V> {
    final int hash;
    final K key;
    V value;
    Node<K, V> next;

    Node(int hash, K key, V value, Node<K, V> next) {
        this.hash = hash;
        this.key = key;
        this.value = value;
        this.next = next;
    }
}

结语

通过深入分析 HashMap 的源码,我们揭示了其底层原理、扩容机制和数据结构。每一行代码都是如何促使 HashMap 这个数据结构在 Java 编程中大放异彩的关键。愿你对 HashMap 有了更深层次的理解,编码的路上更加得心应手!

标签:Node,map,hash,HashMap,剖析,哈希,简单,public,Hashmap
From: https://www.cnblogs.com/lyxlucky/p/17977049

相关文章

  • 深入剖析MyBatis缓存机制
    第1章:引言大家好,我是小黑。今天我们要聊的是MyBatis的缓存机制。作为Java开发中经常使用的持久层框架,MyBatis以其灵活性和简便性而广受欢迎。但你知道吗,很多时候,正是因为这些特点,我们需要更深入地理解它的内部工作原理,尤其是缓存机制。这不仅能帮助我们更高效地使用MyBatis,还能......
  • P4148 简单题 题解
    QuestionP4148简单题有一个\(n\timesn\)的棋盘,每个格子内有一个整数,初始时全部为\(0\),现在需要维护两种操作1xy将格子\(x,y\)里的数字加上\(A\)2x1y1x2y2输出\(x_1,y_1,x_2,y_2\)这个矩形内的数字和强制在线Solution因为强制在线,没法用CDQ什么乱搞,这......
  • Hive SQL底层执行过程详细剖析
     本文结构采用宏观着眼,微观入手,从整体到细节的方式剖析HiveSQL底层原理。第一节先介绍Hive底层的整体执行流程,然后第二节介绍执行流程中的SQL编译成MapReduce的过程,第三节剖析SQL编译成MapReduce的具体实现原理。HiveHive是什么?Hive是数据仓库工具,再具体点就......
  • 简单课程安排问题
    问题描述:假定某大学有门课程需要使用同一个教室来上课。显然,我们不能在一个教室同时上两门或多门课程。因此,每门课使用教室的方式是独享的。假定这n门课程的集合为C={c1,c2,...,cn}。每门课使用教室的时间为{si,fi},i=1,2,...,n。这里si=开始时间,fi=结束时间。假设我们的目标是容纳......
  • 《Java并发实现原理:JDK源码剖析》PDF
    《Java并发实现原理:JDK源码剖析》全面而系统地剖析了JavaConcurrent包中的每一个部分,对并发的实现原理进行了深刻的探讨。全书分为8章,第1章从最基础的多线程知识讲起,理清多线程中容易误解的知识点,探究背后的原理,包括内存重排序、happen-before、内存屏障等;第2~8章,从简单到复杂,逐......
  • 《Java并发实现原理:JDK源码剖析》PDF
    《Java并发实现原理:JDK源码剖析》全面而系统地剖析了JavaConcurrent包中的每一个部分,对并发的实现原理进行了深刻的探讨。全书分为8章,第1章从最基础的多线程知识讲起,理清多线程中容易误解的知识点,探究背后的原理,包括内存重排序、happen-before、内存屏障等;第2~8章,从简单到复杂,逐......
  • 浅谈C++简单前缀和实现
    浅谈前缀和2023.9.28\(tips:\)文章持续更新中,欢迎关注\(upd:\)文章从洛谷博客迁移至博客园(\(2024.1.19\))洛谷B3612【深进1.例1】求区间和题目大E:有一个内部元素个数为\(n\)的数组\(a\),现在有m次询问,求a[l]到a[r]之间所有元素的和朴素的做法:#include<iostream>usin......
  • FFT数据简单校正频率,相位,幅度
    BOOL CWaveBox::DataFftInfo(short*pData,intnCount,FFT_INFO&fft){ Value2D *pIn,*pOut; int i,nPow,nIndex,nAllCount,nRealCount,nHi,nTotal; int TopIndex[FFT_FI_CountMax*2]; Value2D v,v1,v2; double ang1,ang2,ang,dIvt,dBase,dSu......
  • BackgroundWorker 简单示例()
    //异步使用前一定要把下面的代码进行初始化才能成功调用//if(!bgwtemp.IsBusy)//{//bgwtemp.RunWorkerAsync();//}publicBackgroundWorkerbgwtemp=null;publicForm2(){InitializeComponent();bgwtemp=newBackgroundWorker();bgwtemp.DoWork+=bgwtemp_work;......
  • 单机变频器简单启停电路
        西门子参数默认 仅供参考......