Java之HashMap的底层实现

news2024/9/20 13:12:23

Java之HashMap的底层实现

  • 摘要
    • HashMap的底层原理
    • 哈希值转换为数组下标
    • 节点
    • 初始化
    • put(Object key, Object value)
    • 重写toString()
    • get(Object key)
    • 增加泛化
    • remove(K key)

摘要

本博客主要讲述了Java的HashMap的底层实现

HashMap的底层原理

底层原理:数组+链表
在这里插入图片描述
在这里插入图片描述
过程总结:每一个Object的有一个哈希值,通过hashCode()函数获取哈希值,再通过自定义的hash()函数,得到一个值,也就是数组的下标。数组中的每个元素都是一个链表或为空。

哈希值转换为数组下标

在这里插入图片描述

//这就是hash函数,val就是key的哈希值,即val = key.hashCode()
//length 必须是2的整数幂
private int  hash(int val, int length){
       return val & (length - 1);
   }

节点

定义链表中的节点

public class Node2 {
    int hash;//hash对应数组下标
    Object key;
    Object value;
    Node2 next;
}

初始化

//数组元素的类型为Node2
Node2[] table;
int size;

public SxtHashMap02() {
    table = new Node2[16];
}

put(Object key, Object value)

public void put(Object key, Object value){
        Node2 newNode = new Node2();
        newNode.hash = hash(key.hashCode(),table.length);
        newNode.key = key;
        newNode.value = value;
        newNode.next = null;

        Node2 last = null;//这个学习一下,记录最后一个节点
        int index = hash(key.hashCode(),table.length);
        if(table[index] == null){
            table[index] = newNode;
            size ++;
        }else{
            Node2 tmp = table[index];
            while(tmp != null){
                if(key.equals(tmp.key)){
                    System.out.println("key重复了");
                    tmp.value = value;
                    return;
                }else {
                    last = tmp;
                    tmp = tmp.next;
                }
            }
            last.next = newNode;
            size ++;//size的增加与减少不要忘记
        }
    }

重写toString()

public String toString() {
        StringBuilder sb = new StringBuilder();
        sb.append("[");
        for(int i = 0; i < table.length; i ++){
            Node2 temp = table[i];
            while(temp != null){
                sb.append(temp.key + ":" + temp.value + ",");
                temp = temp.next;
            }
        }
        //这个套路学一下,将最后改为']'
        sb.setCharAt(sb.length() - 1,']');
        return sb.toString();
}	

这个toString()有什么用呢?在使用system.out.println()打印的时候,就会用到toString()。

get(Object key)

//根据Map的底层原理,就十分简单
public Object get(Object key){
        int hashCode = key.hashCode();
        int hash = hash(hashCode,table.length);
        Node2 temp = table[hash];
        while(temp != null){
            if(temp.key.equals(key)) return temp.value;
            temp = temp.next;
        }
        return null;
}

增加泛化

public class Node3<K,V> {
    int hash;
    K key;
    V value;
    Node3 next;
}

public class SxtHashMap03<K,V> {
    Node3[] table;
    int size;

    public SxtHashMap03() {
        table = new Node3[16];
    }

    public V get(K key){
        int hashCode = key.hashCode();
        int hash = hash(hashCode,table.length);
        V value = null;
        Node3 temp = table[hash];
        while(temp != null){
            if(temp.key.equals(key)){
                value = (V)temp.value;
            }
            temp = temp.next;
        }
        return value;
    }
    public void put(K key, V value){
        Node3 newNode = new Node3();
        newNode.hash = hash(key.hashCode(),table.length);
        newNode.key = key;
        newNode.value = value;
        newNode.next = null;

        Node3 last = null;
        int index = hash(key.hashCode(),table.length);
        if(table[index] == null){
            table[index] = newNode;
            size ++;
        }else{
            Node3 tmp = table[index];
            while(tmp != null){
                if(key.equals(tmp.key)){
                    System.out.println("key重复了");
                    tmp.value = value;
                    return;
                }else {
                    last = tmp;
                    tmp = tmp.next;
                }
            }
            last.next = newNode;
            size ++;
        }
    }
}

remove(K key)

 public void remove(K key){
        int index = hash(key.hashCode(), table.length);
        Node3 temp = table[index];
        if(temp == null) return;
        if(temp.key.equals(key)){
            table[index] = temp.next;
            size --;
            return;
        }
        Node3 last = null;
        while(temp != null){
            if(temp.key.equals(key)){
                last.next = temp.next;
                size --;
                return;
            }
            last = temp;
            temp = temp.next;
        }
}

参考: 手工实现HashMap

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2058097.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【MySQL】基础入门(第一篇)

目录 一、MySQL的主要特点 二、MySQL的应用场景 三、MySQL的未来发展 四、MySQL的安装 MySQL是一个关系型数据库管理系统&#xff08;RDBMS&#xff09;&#xff0c;由瑞典的MySQL AB公司开发&#xff0c;后来被Sun Microsystems收购&#xff0c;最终在2010年被Oracle公司收…

IDEA安装和使用(配图)

功能强大&#xff1a; 1、强大的整合能力&#xff0c;比如Git,Maven,Spring等 2、开箱即用&#xff08;集成版本控制系统&#xff0c;多语言支持的框架随时可用&#xff09; 3、符合人体工程学 1、高度智能 2、提示功能的快速&#xff0c;便捷&#xff0c;范围广 3、好用…

K8S的统一访问入口-Service

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:Linux运维老纪的首页…

windows本地kafka和zookeeper单机版

文章目录 一、安装zookeeper1.下载zookeeper:[zookeeper下载](https://www.apache.org/dyn/closer.cgi/zookeeper/)2.添加配置文件3.配置windows系统环境变量(可以百度)4.随便打开一个cmd 二、安装kafka1.下载kafka2.修改config目录下的server.properties 三&#xff0c;开始运…

队列的实现和基本操作

队列的表示和实现 <队列是仅在表尾进行插入操作&#xff0c;在表头进行删除操作的线性表&#xff1b; <表尾即an端&#xff0c;表头即a1端&#xff1b;(也称作队尾队头) <它是一种先进先出的(FIFO)线性表 <例如Q(a1,a2,a3,......,an),a1为队头&#xff0c;an…

【中介者模式】设计模式系列:解锁高效协作的秘密武器(设计实战)

文章目录 中介者模式在Java中的应用与实践1. 引言2. 中介者模式解析2.1 模式的基本概念2.2 中介者模式的角色说明2.3 模式的工作原理2.4 UML类图和时序图展示2.5 模式的优点与缺点2.6 模式的变体和扩展 3. 实现细节3.1 Java代码示例3.2 示例应用分析3.3 代码解释 4. 应用场景5.…

css中使用@property自定义属性,实现闪烁渐变背景【2024新属性】

自 2024 年 7 月起&#xff0c;此功能适用于最新的设备和浏览器版本。此功能可能无法在较旧的设备或浏览器中使用。 property 是 CSS 中一个相对较新的功能&#xff0c;主要用于定义自定义属性&#xff08;即 CSS 变量&#xff09;的类型、继承性以及初始值。它允许开发者更好地…

linux使用nginx部署springboot + vue分离项目

第一步,打包后端项目 maven打包springboot项目为jar文件,上传到服务器,然后运行此jar,具体操作参考: centos部署jar包_centos jar 静态资源文件-CSDN博客 第二步,安装nginx 具体操作自行查找,相关命令: 启动: /usr/local/nginx/sbin/nginx 重新加载配置: /usr…

[数据集][目标检测]木材缺陷检测数据集VOC+YOLO格式2383张10类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;2383 标注数量(xml文件个数)&#xff1a;2383 标注数量(txt文件个数)&#xff1a;2383 标注…

PyQt5基础控件

一、按钮 (一)普通按钮QPushButton 1.创建按钮控件 QPushButton()&#xff1a;创建一个无父控件的按钮控件QPushButton(parent)&#xff1a;创建控件的同时, 设置父控件QPushButton(text, parent)&#xff1a;创建控件的同时, 设置提示文本和父控件QPushButton(icon, text, …

自定义组件上传到maven中央仓库2024实测可用最详细版

自 2024 年 3 月 12 日起&#xff0c;官方调整了发布的方式&#xff0c;所有发布都必须通过中央门户&#xff0c;以往老方式可能不适用&#xff0c;以下记录2024新版上传发布方式 注册sonatype账号 Maven中央仓库并不支持直接发布jar包&#xff0c;sonatype是其指定的第三方仓…

【Python】函数进阶(上)

本篇文章将讲解函数进阶的知识&#xff1a; &#xff08;1&#xff09;函数的补充 &#xff08;2&#xff09;函数名是什么 &#xff08;3&#xff09;返回值和print &#xff08;4&#xff09;函数的作用域 1、函数的补充 &#xff08;1&#xff09;参数内存地址相关 如何查…

【算法专题】滑动窗口类

个人主页&#xff1a;CSDN_小八哥向前冲~ 所属专栏&#xff1a;算法基础入门 目录 长度最小的子数组 无重复字符的最长子串 最大连续1的个数 将x减到0的最小操作数 水果成篮 找到字符串中所有字母异位词 最小覆盖字串 长度最小的子数组 题目&#xff1a;【LeetCode】长度…

Python生成432Hz音频

使用 numpy 来生成信号&#xff0c; 使用 matplotlib 可视化信号&#xff0c; 使用 sounddevice 播放声音。 以下生成和播放 432 Hz 的正弦波信号&#xff1a; import numpy as np import sounddevice as sd import matplotlib.pyplot as plt# 生成单音函数 def generate_to…

订单完工数量超过了最大可完工数量

本次完工将造成订单YWS-24070027产出实际完工数量达到了6093.000000000&#xff0c;超过了最大可完工数量5000.000000000 一个成品入库单被它们玩坏了。生产不知道生产了什么。PMC不知道要入库什么。鸡同鸭天天开会之后结果就是单据重复的开立&#xff0c;删除&#xff0c;开立…

C++ TinyWebServer项目总结(5. Linux网络编程基础API)

还是给我的语雀文档打个广告&#xff1a; 《5. C TinyWebServer项目总结&#xff08;5. Linux网络编程基础API&#xff09;》我的文章都是先在语雀里记录的&#xff0c;然后再同步发送到CSDN上&#xff0c;有些格式问题实在是懒得改了&#xff0c;可能会导致大家看的不舒服&…

自行车制造5G智能工厂工业物联数字孪生平台,推进制造业数字化

在当今这个日新月异的数字化时代&#xff0c;制造业正经历着前所未有的变革&#xff0c;自行车制造5G智能工厂工业物联数字孪生平台的兴起&#xff0c;无疑是这场转型浪潮中一股强劲力量。自行车制造5G智能工厂工业物联数字孪生平台的成功应用&#xff0c;不仅仅是技术上的突破…

代码随想录DAY21 - 二叉树 - 08/20

目录 修建二叉搜索树 题干 思路和代码 递归法 迭代法 将有序数组转化为平衡二叉搜索树 题干 思路和代码 递归法 递归优化 迭代法 把二叉搜索树转换为累加树 题干 思路和代码 递归法 迭代法 修建二叉搜索树 题干 题目&#xff1a;给你二叉搜索树的根节点 root …

数据结构【顺序结构二叉树:堆】(1)

​​​​​​​ &#x1f31f;个人主页&#xff1a;落叶 目录 ​ &#x1f525;树的概念与结构​​​​​​​ &#x1f525;树的表⽰ &#x1f525;孩⼦兄弟表⽰法&#xff1a; &#x1f525;树形结构实际运⽤场景 &#x1f525;⼆叉树 &#x1f525;概念与结构 &…

day4JS-数组

1. 什么是数组 数组是值的有序集合。每个值叫做一个元素。每个元素在数组中有一个位置, 以数字表示,称为索引 (有时也称为下标)。数组的元素可以是任何类型。数组索引从 0 开始,数组最大能容纳 4294967295 个元素。 2. 数组的声明与使用 2.1 使用字面量的方式声明数组 语法…