Java 中 Map 集合的基本使用

news2025/1/15 19:45:34

一、HashMap

1.1 HashMap 基本使用

import java.util.*;

/**
 * @author: yunhu
 * @date: 2022/7/14
 */
public class Test {
    public static void main(String[] args) {
        Map<String, Integer> map = new HashMap<>();
        map.put("Alice", 12);
        map.put("Bob", 20);
        map.put("Yunhu", 18);
        for (String key: map.keySet()) {
            Integer value = map.get(key);
            System.out.println("key = " + key + ", value = " + value);
        }
    }
}

output:

key = Yunhu, value = 18
key = Bob, value = 20
key = Alice, value = 12

1.2 特性

  • 非线程安全的
  • 遍历 map不保证有序。
  • 最多允许一个键为 null,值为 null 可以有多个。
  • HashMap 默认的初始化大小为 16。之后每次扩充,容量变为原来的 2 倍。
  • HashMap 中扩容因子的大小是 0.75,当元素到达长度的 75%时,就会进行扩容。

1.3 使用 entrySet() 遍历

import java.util.*;

/**
 * @author: yunhu
 * @date: 2022/7/14
 */
public class Test {
    public static void main(String[] args) {
        Map<String, Integer> map = new HashMap<>();
        map.put("Alice", 12);
        map.put("Bob", 20);
        map.put("Yunhu", 18);
        for (Map.Entry<String, Integer> entry: map.entrySet()) {
            System.out.println("key = " + entry.getKey() + ", value = " + entry.getValue());
        }
    }
}

output:

key = Yunhu, value = 18
key = Bob, value = 20
key = Alice, value = 12

1.4 HashMap 底层原理

1.4.1 获取键的 hashCode

同一个对象没有发生改变,那么他们的 hashCode 是相同的。

1.4.2 获取 hash 值和数组长度

  • 计算 hashCode 的二次 hash 值。
  • 计算出数组长度 len

1.4.3 获取存储地址

index 有两种情况:

  • index 为空,直接存入 value 值。
  • index 不为空,通过 equals判断与已经存在 index 位置的对象是否是同一个对象
    • 如果是,说明值发生更新,覆盖 value
    • 如果不是,发生 hash 冲突,在链表中加入这个对象。

链表的插入情况:在 JDK1.7 以及前是在头结点插入的,在 JDK1.8 之后是在尾节点插入的。

当链表中的元素个数达到 8且数组长度超过 64,链表转为红黑树,查找更快,这个过程叫做「树化」。

当链表中的元素个数小于 6, 红黑树重新转化为链表,这个过程就叫做「链化」。

二、EnumMap

如果 key是一个 enum 枚举类型,那么就可以使用 EnumMap集合,内部使用的是一个数组来存储 value,可以通过枚举的类型来直接定位数组的索引,不需要计算 hashCode()。效率高,并且空间不浪费。

import java.time.DayOfWeek;
import java.util.EnumMap;
import java.util.Map;

/**
 * @author: yunhu
 * @date: 2022/7/14
 */

public class EnumMapTest {
    public static void main(String[] args) {
        Map<DayOfWeek, String> map = new EnumMap<>(DayOfWeek.class);
        map.put(DayOfWeek.MONDAY, "weekday one");
        map.put(DayOfWeek.TUESDAY, "weekday two");
        map.put(DayOfWeek.WEDNESDAY, "weekday three");
        map.put(DayOfWeek.THURSDAY, "weekday four");
        map.put(DayOfWeek.FRIDAY, "weekday five");
        map.put(DayOfWeek.SATURDAY, "weekday six");
        map.put(DayOfWeek.SUNDAY, "weekday seven");
        System.out.println(map);
        System.out.println(map.get(DayOfWeek.FRIDAY));
    }
}

output:

{MONDAY=weekday one, TUESDAY=weekday two, WEDNESDAY=weekday three, THURSDAY=weekday four, FRIDAY=weekday five, SATURDAY=weekday six, SUNDAY=weekday seven}
weekday five

三、TreeMap

3.1 基本使用

TreeMap是有序的,如果键是字符串,那么就按字母顺序输出。

import java.util.*;

/**
 * @author: yunhu
 * @date: 2022/7/15
 */
public class Test {
    public static void main(String[] args) {
        Map<String, Integer> map = new TreeMap<>();
        map.put("yunhu", 1);
        map.put("alice", 2);
        map.put("bob", 3);
        for (String key : map.keySet()) {
            System.out.println(key);
        }
    }
}

output:

alice
bob
yunhu

3.2 自定义排序算法

3.2.1 匿名函数方式

使用 TreeMap必须实现 Comparable接口,StringInteger已经默认实现了,因此可以直接作为键来使用。

如果作为键的类型没有实现 Comparable接口,那么必须指定一个自定义的排序算法。

import java.util.*;

/**
 * @author: yunhu
 * @date: 2022/7/15
 */
public class Test {
    public static void main(String[] args) {
        Map<Person, Integer> map = new TreeMap<>(new Comparator<Person>() {
            public int compare(Person p1, Person p2) {

                // 按年龄从小到大排序
                return p1.age.compareTo(p2.age);
                
                // 按姓名排序
                // return p1.name.compareTo(p2.name);
            }
        });
        map.put(new Person("bob", 10), 1);
        map.put(new Person("alice", 12), 2);
        map.put(new Person("yunhu", 11), 3);
        for (Person key : map.keySet()) {
            System.out.println(key);
        }
    }
}
class Person {
    public String name;
    public Integer age;
    Person(String name, Integer age) {
        this.name = name;
        this.age = age;
    }
    public String toString() {
        return "{Person: " + name + " " + age + "}";
    }
}
{Person: bob 10}
{Person: yunhu 11}
{Person: alice 12}

3.2.2 实现 Comparable 接口方式

import java.util.*;

/**
 * TreeMap 自定义排序方式
 *
 * @author: yunhu
 * @date: 2022/7/15
 */
public class Test {
    public static void main(String[] args) {
        Map<Person, Integer> map = new TreeMap<>();
        map.put(new Person("bob", 10), 1);
        map.put(new Person("alice", 12), 2);
        map.put(new Person("yunhu", 11), 3);
        for (Person key : map.keySet()) {
            System.out.println(key);
        }
    }
}

class Person implements Comparable<Person>{
    public String name;
    public Integer age;
    Person(String name, Integer age) {
        this.name = name;
        this.age = age;
    }
    public String toString() {
        return "{Person: " + name + " " + age + "}";
    }

    @Override
    public int compareTo(Person o) {
        return this.name.compareTo(o.name);
    }
}

output:

{Person: alice 12}
{Person: bob 10}
{Person: yunhu 11}

按姓名排序了。

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

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

相关文章

62.Python 类的3要素语法

62.类的3要素语法 文章目录62.类的3要素语法1. 定义类的语法2. pass语句的作用3.我们学的5个缩进语法4.类的属性5.类的方法6. 创建对象7.类在爬虫中的应用8. 总结1. 定义类的语法 定义类的语法如下&#xff1a; class 类名: 代码块 从语法上来看&#xff1a;类的…

Edify Script (Android Shell)定制Twrp刷机包

本文将通过三方面向大家介绍Edify Script,相信你看完了这篇文章会对安卓系统可刷写插件有一个比较完整的认识,并且能过通过阅读此文章实现使用twrp刷入自己写的插件zipi包。网上能搜到的大部分是旧的,新版本去掉了很多的edify函数,照着他们那个写多半会报错。 本文分为如下…

LeetCode学习-第三十八天

第三十八天 我使用的C&#xff0c;错误的地方请见谅&#xff0c;文章初衷仅用来督促本人学习&#xff0c;如果恰巧能够给你带来帮助&#xff0c;我会十分开心。 文章目录第三十八天一、537. 复数乘法二、29. 两数相除一、537. 复数乘法 复数 可以用字符串表示&#xff0c;遵循…

CentOS7如何修改IP地址及UUID

CentOS7系统下&#xff0c;如果要把IP地址修改为192.168.1.80&#xff0c;子网掩码是255.255.255.0&#xff0c;网关是192.168.1.1&#xff0c;DNS是192.168.1.1&#xff0c;那么我们使用客户端连接CentOS或者打开系统终端&#xff0c;切换到root用户&#xff0c;命令&#xff…

开料的目的和子流程,一文读懂

经过多个月的分享&#xff0c;关于PCB行业&#xff0c;想必朋友们已经有了一些个人的理解&#xff0c;甚至对PCB行业&#xff0c;还产生了浓厚的兴趣。 但是&#xff0c;PCB生产工艺是非常复杂的&#xff0c;想要深入地学习并且学好PCB生产工艺&#xff0c;假如不在PCB的生产一…

如何从PyTorch中获取过程特征图

一、获取Tensor 神经网络在运算过程中实际上是以Tensor为格式进行计算的&#xff0c;我们只需稍稍改动一下forward函数即可从运算过程中抓到Tensor 代码如下&#xff1a; base_feature self.extractor.forward(x) #正常的前向传递 featurebase_feature.detach() …

JavaScript 数据类型

文章目录JavaScript 数据类型JavaScript 拥有动态类型JavaScript 字符串JavaScript 数字JavaScript 布尔JavaScript 数组JavaScript 对象Undefined 和 Null声明变量类型JavaScript 数据类型 字符串&#xff08;String&#xff09;、数字(Number)、布尔(Boolean)、数组(Array)、…

基于java eclipse+jsp+mysql+servlet+Spring的学生信息管理系统基础版

基于java eclipsejspmysqlservletSpring的学生信息管理系统基础版 博主介绍&#xff1a;5年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 超级帅帅吴 Java毕设项目精品实战案例《500套》 欢迎点赞 收藏 ⭐留言…

【Python百日进阶-数据分析】Day148 - plotly直方图:px.histogram()

文章目录四、实例4.1 带有 Plotly Express 的直方图4.1.1 基本直方图4.1.2 使用一个包含分类数据的列4.1.3 选择方箱的数量4.1.4 日期数据直方图4.1.5 分类数据的直方图4.1.6 访问计数&#xff08;y 轴&#xff09;值4.1.7 标准化类型4.1.8 直方图的外观4.1.9 一列不同值的几个…

MySQL基础篇第10章(创建和管理表)

1. 基础知识 1.1 一条数据存储的过程 存储数据是处理数据的第一步 。只有正确地把数据存储起来&#xff0c;我们才能进行有效的处理和分析。否则&#xff0c;只 能是一团乱麻&#xff0c;无从下手。 那么&#xff0c;怎样才能把用户各种经营相关的、纷繁复杂的数据&#xff…

npm 包 chalk-next 被开发者投毒,源码 SRC 目录被删除

一、事件简述 1月5日&#xff0c;有开发者在 twitter 中发文称遭遇了名为 chalk-next 的组件投毒事件&#xff0c;该组件存在收集配置信息和删除本地文件的恶意逻辑&#xff0c;当前 NPM 仓库已经下线了该组件。 chalk-next 组件的开发者也是 vue-admin-beautiful 项目的作者 …

配对交易策略设计

可交易的一个关键要求是两个股票的对数价格序列之间存在均衡关系。 而均衡关系由两个值描述&#xff1a;协整系数以及均衡值。一旦确定这两个值&#xff0c;它们就可以用来构建两种股票的对数价格的线性组合&#xff0c;即所谓的价差。配对交易是对价差均值回归特性的押注。当…

Gigabyte B450 Aorus Elite AMD Ryzen 5 3600电脑 Hackintosh 黑苹果efi引导文件

硬件型号驱动情况主板Gigabyte B450 Aorus Elite处理器AMD Ryzen 5 3600已驱动内存16GB Samsung DDR3 1600MHz已驱动硬盘Samsung 840 Evo 250GB已驱动显卡AMD Radeon RX570 4gb已驱动声卡Realtek ALC892已驱动网卡Realtek 8111G已驱动无线网卡蓝牙Asus AX3000 Dual Band PCI-E …

(一)云原生基础概念-容器-Kubernets

文章目录云原生云原生历程云原生技术范畴云原生与传统部署云原生关键技术传统vs微服务vsServerlessServerless容器分层复用模式容器运行构建镜像容器引擎架构-moby容器和VM之间的差异Kubernetes调度自动恢复水平伸缩总架构Master架构Node架构Pod一个kubernet架构VolumeDeployme…

【计组】GPU--《深入浅出计算机组成原理》(六)

​课程链接&#xff1a;深入浅出计算机组成原理_组成原理_计算机基础-极客时间 GPU 是随着计算机里面需要渲染三维图形的出现&#xff0c;而发展起来的设备。 一、GPU和图形渲染 完整的5 个步骤的渲染流程一般也被称为图形流水线&#xff08;Graphic Pipeline&#xff09; …

SREWorks v1.4 版本发布 | 离线安装 前端重构

在 v1.3 版本之后&#xff0c;SREWorks 团队收集了较多的用户反馈&#xff0c;大家普遍对于 SREWorks 的内网离线安装有较大的诉求。于是团队决定进一步增强这部分的安装能力。 前端工程部分 (frontend)&#xff0c;为了开发者更加敏捷高效的协作开发&#xff0c;以及便于社区…

web(一)—— HTML基础(web标准、开发工具、标签)

目标能够理解HTML的 基本语法 和标签的关系 能够使用 排版标签 实现网页中标题、段落等效果 能够使用 相对路径 选择不同目录下的文件 能够使用 媒体标签 在网页中显示图片、播放音频和视频 能够使用 链接标签 实现页面跳转功能一、基础认知目标&#xff1a;认识 网页组成 和 五…

vite使用css的各种功能

1.使用公共的变量(:root)定义在root的变量可以进行类的使用 :root { font-size: 160px;--main-bg-color:red } .red{color:var(--main-bg-color) }你看这个hello world变得多大多红 2.vite使用postcss-plugins/console’ 2-1安装&#xff1a;cnpm install postcss-plugins/co…

最大公约数-欧几里得算法

最近在复习数论&#xff0c;欢迎来到数论的起点gcd 最大公因数&#xff0c;也称最大公约数、最大公因子&#xff0c;指两个或多个整数共有约数中最大的一个。 欧几里得算法&#xff08;辗转相除法&#xff09; 已知两个数a和b&#xff0c;求出两数的最大公约数首先证明: 不妨…

uni-app入门:组件的基本使用

1.组件概念 2.组件分类 2.1 基础组件 2.2 拓展组件 2.3 easycom规范 3.自定义组件以及使用 3.1局部注册 3.2全局注册 1.组件概念首先讲一下什么是组件 官方说法&#xff1a; 组件是视图层的基本组成单元。 组件…