Java-数据结构(二)-Map:HashMap、TreeMap、LinkedHashMap

news2024/11/17 6:33:23

目录

  • 一、 引言
  • 二、问题
    • 2.1 什么是Map
    • 2.2 使用Map的好处
    • 2.3 Map的底层原理
    • 2.4 Key和Value的含义
    • 2.5 Key值为什么不能重复
    • 2.6 Key值和Hash的关系
  • 三、 HashMap
    • 3.1 初始化HashMap
    • 3.2 添加和获取元素
    • 3.3 遍历HashMap
    • 3.4 删除元素
    • 3.5实现原理
      • ①HashMap的put()方法
      • ②HashMap的get()方法
  • 四、 TreeMap
    • 4.1 初始化TreeMap
    • 4.2 添加和获取元素
    • 4.3 遍历TreeMap
    • 4.4 删除元素
  • 五、 LinkedHashMap
    • 5.1 初始化LinkedHashMap
    • 5.2 添加和获取元素
    • 5.3 遍历LinkedHashMap
    • 5.4 删除元素
  • 六、二维表总结

一、 引言

    Map是Java中常用的数据结构,它提供了一种键值对的存储方式,可以根据键来快速访问值。在本篇文章中,我将学习Java中的Map数据结构

    问题是最好的老师,我将从至少以下几个方面阐述,什么是map、使用Map有什么好处、Map的底层原理、map中的key和value分别是什么、以及Map的Key值为什么不能重复、Map中的key值和Hash有什么关系。

    以及对HashMap、TreeMap和LinkedHashMap三种常用的Map实现类进行了解。我将逐步解析它们的初始化、添加和获取元素、遍历和删除元素等功能,最后给出一个二维表进行结构化。

二、问题

2.1 什么是Map

    Map是Java中的一个接口,它代表了一种键值对的映射关系。它允许我们通过Key来访问Value。在Map中,每个Key都是唯一的,而且与该Key对应的Value是一一对应的关系。

2.2 使用Map的好处

    使用Map有很多好处,以下是其中几个重要的好处:

    快速访问:通过给定的Key,可以快速地访问对应的Value,无需遍历整个集合。
    灵活性:Map不仅可以存储基本数据类型的值,还可以存储自定义对象作为Value,这使得它非常灵活。
    动态增长:Map的大小可根据需要动态增长,不需要事先指定容量。
    数据分类:Map可以用于对数据进行分类、分组和存储,为后续的检索和处理提供了便利。

2.3 Map的底层原理

    在Java中,常用的Map实现类有HashMap、TreeMap和LinkedHashMap。这些实现类在底层的数据组织方式和查找算法上略有不同。

    HashMap使用散列表(Hash Table)作为底层数据结构,它通过把Key的Hash值映射到一个数组索引上,并使用链地址法解决Hash冲突。

    TreeMap使用红黑树(Red-Black Tree)作为底层数据结构,它会对Key进行排序,并且可以提供有序的遍历。

    LinkedHashMap继承自HashMap,它在HashMap的基础上通过维护一个双向链表来保证插入顺序或访问顺序。

    根据实际情况选择不同的Map实现类,可以根据需求来平衡时间复杂度和空间复杂度。

    下面将会了解这几个实现类的一些方法。

2.4 Key和Value的含义

在Map中,Key用于唯一标识一个键值对,它相当于数据的索引。Value则是与Key相关联的数据。对于同一个Key,只能有一个对应的Value,但是不同的Key可以对应不同的Value。

例如,我们可以创建一个Map,将每个人的名字作为Key,将他们的年龄作为Value。通过Key,我们可以快速地查找到对应的年龄。

2.5 Key值为什么不能重复

    Map中要求每个Key都是唯一的,这是因为Map需要通过Key来定位和访问Value。如果出现多个相同的Key,Map无法确定应该返回哪个Value。

    当我们使用put方法向Map中添加键值对时,如果Key已经存在,新的Value将会覆盖旧的Value。因此,Key的唯一性保证了在Map中定位Value的准确性。

2.6 Key值和Hash的关系

    Java中的HashMap和LinkedHashMap是通过计算Key的Hash值来确定Key在底层数组中的位置的。

    在HashMap中,当我们向其中插入一个键值对时,HashMap会首先通过Key的hashCode()方法计算Key的哈希值。然后,HashMap会根据哈希值对数组的长度取模,得到Key在底层数组中的索引位置。如果有多个Key的哈希值映射到同一个索引位置,则HashMap会使用链表或红黑树来处理冲突。

    而在LinkedHashMap中,它在HashMap的基础上通过维护一个双向链表来保证插入顺序或访问顺序。HashMap中的Key与链表节点相互关联,实现了按照插入顺序或访问顺序迭代Map的键值对。

    由此可见,Hash在Map中起到了定位Key的作用,通过计算Key的哈希值,可以快速地定位到Key在底层数组中的位置,从而提高了查找效率。同时,Hash值的唯一性也保证了Key在Map中的唯一性。

    由于Hash值是通过哈希函数计算得出的,存在一定的碰撞概率。因此,在使用自定义对象作为Key时,我们需要重写hashCode()方法来确保生成的Hash值能够准确地表示对象的唯一性,同时也要重写equals()方法来处理碰撞冲突时的比较逻辑。这样可以保证不同的Key对象即使在Hash值相同的情况下,也可以正确地进行比较和查找。

三、 HashMap

在这里插入图片描述

3.1 初始化HashMap

    在Java中,我们可以使用HashMap类来创建一个HashMap对象。下面是一些常见的初始化方法:

    使用默认构造函数:

HashMap<String, Integer> map = new HashMap<>();

    指定初始容量:

HashMap<String, Integer> map = new HashMap<>(16);

指定初始容量和加载因子:

HashMap<String, Integer> map = new HashMap<>(16, 0.75f);

3.2 添加和获取元素

    向HashMap中添加元素时,我们需要使用put()方法,并提供键和值。下面是一个示例:

HashMap<String, Integer> map = new HashMap<>();
map.put("apple", 5);
map.put("banana", 10);
map.put("orange", 8);

    获取HashMap中的元素可以使用get()方法,并提供键。下面是一个示例:

int appleCount = map.get("apple");
System.out.println("苹果数量:" + appleCount);

3.3 遍历HashMap

    遍历HashMap可以使用多种方式,比如使用迭代器、for-each循环或使用Java 8的Lambda表达式。下面是一个使用迭代器遍历HashMap的示例:

Iterator<Map.Entry<String, Integer>> iterator = map.entrySet().iterator();
while (iterator.hasNext()) {
    Map.Entry<String, Integer> entry = iterator.next();
    System.out.println("水果:" + entry.getKey() + ",数量:" + entry.getValue());
}

3.4 删除元素

    从HashMap中删除元素可以使用remove()方法,并提供键。下面是一个示例:

map.remove("banana");

3.5实现原理

①HashMap的put()方法

实现原理如下:

    首先,HashMap将要存储的键值对通过哈希函数进行处理,得到一个哈希码(hash code)。

    接下来,HashMap将根据哈希码找到该键值对应在内部数组中的索引位置。

    如果该位置上没有其他键值对存在,那么直接将新的键值对存储在该位置上即可。

    如果该位置上已经存在其他键值对,那么HashMap将采用链表或者红黑树的方式来处理冲突。它会在该位置上的键值对链表(或树)中依次比较存储的键的哈希码和键值是否与要存储的键值对相等。

    如果找到了相等的键,HashMap会替换该键对应的值。

    如果没有找到相等的键,HashMap会将新的键值对添加到链表(或树)的末尾。

在这里插入图片描述

②HashMap的get()方法

它的实现原理如下:

    首先,HashMap通过哈希函数计算键的哈希码。

    接下来,HashMap将根据哈希码找到该键对应在内部数组中的索引位置。

    如果该位置上没有键值对,那么表示该键不存在于HashMap中,返回null。

    如果该位置上存在键值对,HashMap会遍历链表(或树),比较存储的键的哈希码和键值是否与要获取的键值对相等。

    如果找到了相等的键,HashMap返回该键对应的值。

    如果遍历完链表(或树)都没有找到相等的键,那么表示该键不存在于HashMap中,返回null。

    通过这种方式,HashMap可以高效地实现put()和get()方法,快速存储和查找键值对,提供了快速的数据访问能力。

四、 TreeMap

在这里插入图片描述

4.1 初始化TreeMap

    与HashMap类似,我们可以使用TreeMap类来创建一个TreeMap对象。下面是一些常见的初始化方法:

    使用默认构造函数:

TreeMap<String, Integer> map = new TreeMap<>();

    使用Comparator初始化:

TreeMap<String, Integer> map = new TreeMap<>(Comparator.reverseOrder());

4.2 添加和获取元素

    添加和获取元素的方式与HashMap类似,使用put()方法添加元素,使用get()方法获取元素。

map.put("apple", 1);
map.put("banana", 2);
map.put("orange", 3);

int value = map.get("apple");
System.out.println(value);  // 输出:1

4.3 遍历TreeMap

    遍历TreeMap的方法与HashMap类似,可以使用迭代器、for-each循环或Lambda表达式。

    以下是使用for-each循环遍历TreeMap的例子:

for (Map.Entry<String, Integer> entry : map.entrySet()) {
    String key = entry.getKey();
    int value = entry.getValue();
    System.out.println(key + " : " + value);
}

4.4 删除元素

    从TreeMap中删除元素的方式与HashMap类似,使用remove()方法。

map.remove("banana");

五、 LinkedHashMap

在这里插入图片描述

5.1 初始化LinkedHashMap

    LinkedHashMap是一个有序的Map实现类,保留了元素的插入顺序。初始化方式与HashMap类似。

    使用默认构造函数:

LinkedHashMap<String, Integer> map = new LinkedHashMap<>();

5.2 添加和获取元素

    添加和获取元素的方式与HashMap类似,使用put()方法添加元素,使用get()方法获取元素。

map.put("apple", 1);
map.put("banana", 2);
map.put("orange", 3);

int value = map.get("apple");
System.out.println(value);  // 输出:1

5.3 遍历LinkedHashMap

    遍历LinkedHashMap的方法与HashMap类似,可以使用迭代器、for-each循环或Lambda表达式。

for (Map.Entry<String, Integer> entry : map.entrySet()) {
    String key = entry.getKey();
    int value = entry.getValue();
    System.out.println(key + " : " + value);
}

5.4 删除元素

    从LinkedHashMap中删除元素的方式与HashMap类似,使用remove()方法。

map.remove("banana");

六、二维表总结

维度HashMapTreeMapLinkedHashMap
底层实现哈希表红黑树哈希表+链表
插入顺序无序无序(基于键的自然排序或自定义排序)保持插入顺序
查找效率O(1)O(log n)O(1)
迭代顺序无序有序(基于键的自然排序或自定义排序)保持插入顺序或访问顺序
键的唯一性允许null键和null值允许null键和null值允许null键和null值
性能在大多数情况下,具有良好的性能相比HashMap,由于排序逻辑,稍稍慢一些相比HashMap,由于维护链表,稍稍慢一些
空间需求相对较低(无序)相对较高(有序)相对较高(保持插入顺序或访问顺序)

    我们可以看到HashMap、TreeMap和LinkedHashMap都是非常有用和灵活的Map实现类,每种都适用于不同的使用场景。当我们需要一个无序、高效的Map时,可以选择HashMap;当我们需要一个有序的Map时,可以选择TreeMap;而当我们需要一个保留插入顺序、支持特殊操作的Map时,可以选择LinkedHashMap。

    因此,当我们需要使用Map数据结构时,我们可以根据具体需求选择合适的数据结构来解决问题。

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

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

相关文章

【海量数据挖掘/数据分析】 之 贝叶斯分类算法(朴素贝叶斯分类、贝叶斯分类计算流程、拉普拉斯修正、贝叶斯分类实例计算)

【海量数据挖掘/数据分析】 之 贝叶斯分类算法&#xff08;朴素贝叶斯分类、贝叶斯分类计算流程、拉普拉斯修正、贝叶斯分类实例计算&#xff09; 目录 【海量数据挖掘/数据分析】 之 贝叶斯分类算法&#xff08;朴素贝叶斯分类、贝叶斯分类计算流程、拉普拉斯修正、贝叶斯分类…

无java环境运行jar

1、编写简单java程序。 例&#xff1a; public static void main(String[] args) {if(args.length>0)System.out.println("Params is&#xff1a;"args[0]);System.out.println("Hello word ! I am demo&#xff0c;&#xff0c;&#xff0c;&#xff0c…

软考高级网规考试笔记(涉及表格用图片代替_9万字左右)

作者&#xff1a;BSXY_19计科_陈永跃_23年7月更 BSXY_信息学院_v:CwJp0403 注&#xff1a;未经允许禁止转发任何内容 笔记说明&#xff1a; 目前还只有笔记&#xff0c;其他资源将会在近期更新&#xff0c;&#xff08;笔记5.5即可&#xff0c;不诚勿加可以去其他地方自找找看&…

Quartz任务调度笔记

一、概念 1.1简介 Quzrtz是OpenSymphony开源组织在Job scheduling领域的开源项目 特点&#xff1a;Quartz具有丰富特性的"任务调度库"&#xff0c;能够集成于任何的Java应用&#xff0c;小到独立的应用&#xff0c;大到电子商业系统。quartz能够创建亦简单亦复杂的调…

DAY35:贪心算法(二)分发饼干+摆动序列

文章目录 455.分发饼干思路两个for循环嵌套的写法为什么这种写法必须要有visited数组debug测试逻辑问题&#xff1a;没有进行计数逻辑问题&#xff1a;找到了result3个孩子 一层for循环的写法为什么这种写法一定要把小孩数组放在外面 376.摆动序列&#xff08;逻辑问题&#xf…

02_04实时调度类及SMP多核处理器的实时操作系统体系结构

上一篇文章说的是普通进程的调度但同时还有实时进程在linux上面进行运行 这边来看看实时进程在linux里面怎么调度 同时linux操作系统对实时任务的处理方式和设计思想 实时调度类 Linux进程分为两大类:实时进程和普通进程。 实时进程与普通进程根本不同之处&#xff0c;如果系…

ModuleNotFoundError: No module named ‘transformers_modules.chatglm2-6b‘解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

青少年机器人技术一级核心知识点:机械结构及模型(四)

随着科技的不断进步&#xff0c;机器人技术已经成为了一个重要的领域。在这个领域中&#xff0c;机械结构是机器人设计中至关重要的一部分&#xff0c;它决定了机器人的形态、运动方式和工作效率。对于青少年机器人爱好者来说&#xff0c;了解机械结构的基础知识&#xff0c;掌…

LabVIEW大模拟数据解决方案

LabVIEW大模拟数据解决方案 比亚迪汽车对于在动力总成标定和控制部门工作的400多名工程师来说&#xff0c;这种投资包括实现NI的新战略和解决方案&#xff0c;以更好地捕获和管理大量的原始测试数据&#xff0c;在车辆上市前做出更明智的决策。 因为能够更好地获取更优质的数…

Go语言开发者的Apache Arrow使用指南:内存管理

如果你看了上一篇《Go语言开发者的Apache Arrow使用指南&#xff1a;数据类型》[1]中的诸多Go操作arrow的代码示例&#xff0c;你很可能会被代码中大量使用的Retain和Release方法搞晕。不光大家有这样的感觉&#xff0c;我也有同样的feeling&#xff1a;**Go是GC语言[2]&#x…

MWCS 2023,到底有些啥?(下篇)

█ 亚信科技 5G行业专网一体机&#xff1a; 反光太厉害了&#xff0c;看不太清&#xff1a; 这几张都是小枣妹拍的&#xff0c;^_^&#xff1a; █ 浩鲸科技 浩鲸&#xff0c;就是以前的中兴软创&#xff1a; █ 紫光展锐 6G这块&#xff0c;干货很多&#xff1a; 这次重点展示…

docker容器日志占满硬盘空间的解决方案

目录 原因分析解决方案方案一 定时清空日志文件方案二 全局容器日志大小方案三 修改日志驱动 docker常用清理空间命令 原因分析 由于默认情况下&#xff0c;docker使用json-file类型的日志驱动&#xff0c;该日志驱动默认情况下&#xff0c;每个容器的日志会一直追加在文件名为…

chatgpt赋能python:用Python模拟用户登录,实现多个网站的SEO优化

用Python模拟用户登录&#xff0c;实现多个网站的SEO优化 介绍 在互联网时代&#xff0c;SEO已成为许多网站提高曝光率和流量的重要手段之一。而SEO优化的一个重要方面就是网站的用户登录。然而&#xff0c;手动登录多个网站进行SEO操作是非常耗时耗力的。那么&#xff0c;有…

卷积神经网络实现猫狗分类

目录 一、环境配置二、神经网络CNN1、简介2、CNN结构3、层次说明 三、数据集准备1、下载数据集2、数据集分类 四、 猫狗分类的实例——基准模型1、构建网络模型2、配置训练方法3、转换格式4、模型训练并保存生成的模型5、结果可视化 五、调整基准模型1、图像增强2、增强后的图像…

【C语言初阶(8)】函数1

文章目录 1. 函数的介绍2. 函数的分类2.1 库函数2.2 自定义函数 3. 函数的参数4. 函数的调用4.1 传值调用4.2 传址调用 1. 函数的介绍 1. 什么是函数&#xff1f; 函数是完成特定任务的独立程序代码单元。语法规则定义了函数的结构和使用方式。 一些函数执行某些动作&#xff…

Guava 之 EventBus

​​EvenBus​​​ 是 Guava 中 Pub/Sub 模式的轻量级实现。平时开发如果我们要实现自己的 Pub/Sub 模型&#xff0c;要写不少类&#xff0c;设计也挺复杂&#xff0c;对业务代码也有一定的侵入&#xff0c;但是在使用了 ​​EventBus​​ 之后就很方便了。 在 Pub/Sub 模式中…

Java——《面试题——tomcat篇》

全文章节 Java——《面试题——基础篇》 Java——《面试题——JVM篇》 Java——《面试题——多线程&并发篇》 Java——《面试题——Spring篇》 Java——《面试题——SpringBoot篇》 Java——《面试题——MySQL篇》​​​​​​ Java——《面试题——SpringCloud》 Java——…

干货 | 智慧教育平台生成式人工智能应用的安全要求

以下内容整理自清华大学《数智安全与标准化》课程大作业期末报告同学的汇报内容。 第一部分&#xff1a;编制说明 标准制定的基本原则主要包括以下四个方面&#xff1a; 综合性&#xff1a;本标准全面漫盖了智慧教育平台ChatGPT安全保护的要求&#xff0c;以便用户参考&#xf…

Spring:Bean

Bean 概述配置方式自动装配继承与依赖作用域外部属性文件的使用 概述 Spring 容器负责管理依赖注入&#xff0c;它将被管理的对象都称为 bean 。我们通过 xml 文件配置方式进行对 bean 的声明和管理。 写法如下&#xff1a; <beans><bean id"bean的唯一标识符…

Scrapy框架--CrawlSpider (详解+例子)

目录 CrawlSpider 简介 基本运行 特性和概念 基本使用 创建CrawlSpider 运行 使用CrawlSpider中核心的2个类对象 Rule对象 LinkExtractors 作用 使用 查看效果-shell中验证 示例 注意 CrawlSpider 简介 CrawlSpider 是 Scrapy 框架提供的一个特殊的 Spider 类…