Java中的映射Map - 入门篇
前言
大家好啊,我是汤圆,今天给大家带来的是《Java中的映射Map - 入门篇》,希望对大家有帮助,谢谢
简介
前面介绍了集合List,这里开始简单介绍下映射Map,相关类如下图所示

正文
Map是一种存储键值对的数据集合,键以散列或者树形结构进行存储;
为什么会设计Map接口呢?
假设我们有一个员工类,里面有Id属性和姓名等其他信息,现在我们把所有员工都存到List里,然后要找出Id为001的员工,你会发现,你需要从List中遍历每个对象,然后取出Id进行比较;
你会发现这种查找法效率很低,有点杀鸡用牛刀的感觉;
这时如果有一个集合类,可以以键值对映射的方式的存储员工信息(Id作为键,员工信息作为值),那么就可以只遍历键列表,然后进行比较;
你会发现这种查找法效率提高了很多,因为物尽其用了(比较的是Id,也只是取了Id,没有浪费);
这就是Map接口的作用,可以根据某个键去查找对应的信息,有点类似于数据库的设计。
Map的种类
Map主要有三种类型:HashMap(常用)、TreeMap(树形结构)、LinkedHashMap(前两者的结合)
我们先来看一下Map接口主要的几个方法:
V put(K key, V value):往Map中添加键值对,其中key为键,value为值;如果key存在,则覆盖原有的值;如果不存在,则新建键值对。V get(Object key):从Map中查找键key对应的值,如果没有,则返回nulldefault V getOrDefault(Object key, V defaultValue):从Map中查找键key对应的值,如果没有,则返回第二个参数(设置的默认值);这里的修饰符default是用在接口方法中,表示这个方法在接口中已经实现了,子类可以不实现(Java8开始支持)Set<K> keySet():返回Map中Key的集合;之所以返回Set,是因为Map中的key不能有重复,所以用Set最适合了Collection<V> values():返回Map中Values的集合
下面我们简单看下三者的区别
| HashMap | TreeMap | LinkedHashMap | |
|---|---|---|---|
| 访问速度 | 快 | 慢 | 适中 |
| 元素是否有序 | 无序 | 有序,默认按key排序 | 有序,默认按插入的顺序 |
| 适用场景 | 普通的插入,查询(用的最多) | 需要对key进行排序的场景(比如员工按年龄排序等) | 需要保证查询和插入顺序一致的场景(类似队列) |
接下来我们以HashMap为例,来介绍Map接口
HashMap
HashMap内部是数组+链表的结构;
因为在添加键值对的时候,Key做了hash处理,然后按照hash值进行排列;
- 如果hash值没有重复,就按照数组的方式依次排列;
- 如果hash值有重复的,就添加到已有的键值对后面(Java8以后是尾部插入),形成链表结构;
整体结构 如下图所示

这里只是简单介绍,以后再深入了解
下面用代码示范一下
// 键值对集合,键不可以重复
Map<String, Integer> map = new HashMap<>();
// 添加:首先会检查对应的key是否存在,如果不存在,则新建键值对,然后填充;如果存在,则覆盖已有的值
map.put("a", 1); // 这里的1会自动装箱为Intege类型
// 查询
int value1 = map.get("a");
int value2 = map.get("b");
System.out.println(map);
这里有个很有意思的现象,你觉得value2会是多少呢?
答案是多少都不是,因为程序运行到这一行就出错了,报空指针异常

不应该返回null吗?怎么会出错?
这里涉及到拆箱和装箱的问题,上面我们在添加put的时候,int 1自动装箱为Integer;
然后在获取get的时候,对应的也是要进行拆箱的,将Integer转为int;
但是由于获取的value = null,所以就相当于对null进行拆箱,结果就报错了。
解决办法就是严格按照Map的类型信息进行添加和获取;
将上面的代码加以修改,如下所示
// 键值对集合,键不可以重复
Map<String, Integer> map = new HashMap<>();
// 添加:首先会检查对应的key是否存在,如果不存在,则新建键值对,然后填充;如果存在,则覆盖已有的值
map.put("a", 1); // 这里的1会自动装箱为Intege类型
// 查询
Integer value1 = map.get("a");
Integer value2 = map.get("b");
System.out.println(map);
此时value2就等于null了。
关于自动装箱和拆箱,网上资源很多,这里就不再细说了
TreeMap
TreeMap在插入的时候,可以按照键的顺序进行排序
它适合用在排序比较多的场景,性能会比HashMap差一些
LinkedHashMap
LinkedHashMap拥有HashMap的大部分优点,且保证了插入的顺序,使得在查询的时候,可以按照插入的顺序依次读取
三者的排序比较
下面用代码演示一下,依次插入100个数,看看他们分别是怎么排序的
HashMapDemo.java
public class HashMapDemo {
public static void main(String[] args) {
// 键值对集合,键不可以重复
Map<String, Integer> map = new HashMap<>();
// 倒序插入100个数
int i =100;
while (i-->0){
map.put(""+i, i);
}
// 查询
for (String str :
map.keySet()) {
// 这里会乱序输出
System.out.println(str);
}
}
}
输出如下所示:很乱

TreeMapDemo.java
public class MapDemo {
public static void main(String[] args) {
// TreeMap
Map<String, Integer> map1 = new TreeMap<>();
// 连续倒序插入100个数
int k =100;
while (k-->0){
map1.put(""+k, k);
}
// 查询
for (String str :
map1.keySet()) {
// 这里会正序输出
System.out.println(str);
}
}
}
输出如下所示:

细心的你们,应该会发现上面的输出有点别致
那是因为这里的键key(0~99)其实不是整型,而是字符串类型,所以排序按照字符串的升序来排,才会出现如图所示的结果
(建议实际场景不要这样搞,容易出事,字符串尽量不要用纯数字,而是要跟字母做拼接;)
正确的做法是key=“a”+i,这种方式
LinkedHashMapDemo.java
public class MapDemo {
public static void main(String[] args) {
// LinkedHashMap
Map<String, Integer> map2 = new LinkedHashMap<>();
// 倒序插入100个数
int j =100;
while (j-->0){
map2.put("a"+j, j);
}
for (String str :
map2.keySet()) {
// 这里按照插入的顺序依次输出
System.out.println(map2.get(str));
}
}
}
输出如下所示:

总结
Map一般用到的有HashMap,TreeMap,LinkedHashMap,当然还有并发相关的,这里入门级别的先不涉及(比如ConcurrentHashMap)
HashMap的插入和访问都很快,但是内部是无序排列
TreeMap的插入和访问都很慢,但是内部是有序排列,默认按key升序排列
LinkedHashMap拥有HashMap的大部分优点,而且还可以按照元素插入的顺序来访问元素,但是性能会比HashMap差
后记
最后,感谢大家的观看,谢谢
Java中的映射Map - 入门篇的更多相关文章
- Java中的集合Set - 入门篇
前言 大家好啊,我是汤圆,今天给大家带来的是<Java中的集合Set - 入门篇>,希望对大家有帮助,谢谢 简介 前面介绍了集合List,映射Map,最后再简单介绍下集合Set,相关类如下 ...
- Java中的IO流 - 入门篇
前言 大家好啊,我是汤圆,今天给大家带来的是<Java中的IO流-入门篇>,希望对大家有帮助,谢谢 由于Java的IO类有很多,这就导致我刚开始学的时候,感觉很乱,每次用到都是上网搜,结果 ...
- Java中的集合List - 入门篇
前言 大家好啊,我是汤圆,今天给大家带来的是<Java中的集合List - 入门篇>,希望对大家有帮助,谢谢 简介 说实话,Java中的集合有很多种,但是这里作为入门级别,先简单介绍第一种 ...
- Java中如何遍历Map对象的4种方法
在java中遍历Map有不少的方法.我们看一下最常用的方法及其优缺点. 既然java中的所有map都实现了Map接口,以下方法适用于任何map实现(HashMap, TreeMap, LinkedHa ...
- 转!! Java中如何遍历Map对象的4种方法
在Java中如何遍历Map对象 How to Iterate Over a Map in Java 在java中遍历Map有不少的方法.我们看一下最常用的方法及其优缺点. 既然java中的所有map都 ...
- 【转】Java中如何遍历Map对
在Java中如何遍历Map对象 How to Iterate Over a Map in Java 在java中遍历Map有不少的方法.我们看一下最常用的方法及其优缺点. 既然java中的所有map都 ...
- 【转】Java中如何遍历Map对象的4种方法
原文网址:http://blog.csdn.net/tjcyjd/article/details/11111401 在Java中如何遍历Map对象 How to Iterate Over a Map ...
- (转)在Java中如何遍历Map对象
在Java中如何遍历Map对象 How to Iterate Over a Map in Java 在java中遍历Map有不少的方法.我们看一下最常用的方法及其优缺点. 既然java中的所有map都 ...
- (转载)Java中如何遍历Map对象的4种方法
在Java中如何遍历Map对象 How to Iterate Over a Map in Java 在java中遍历Map有不少的方法.我们看一下最常用的方法及其优缺点. 既然java中的所有map都 ...
随机推荐
- macOS warning emoji render bug
macOS warning emoji render bug ️ macOS render bug Apple Color Emoji fonts install old version fonts ...
- 手机 wifi 已连接,不可上网 bug
手机 wifi 已连接,不可上网 bug 同一个 Wi-Fi,电脑却可以? 注意事项 Mac 共享热点支持有线连接.蓝牙连接的网络进行共享. 如果你的 Mac 本身是通过 wifi 来连接上网的,那就 ...
- js & while & do while
js & while & do while https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Stat ...
- [Python学习笔记]文件的读取写入
文件与文件路径 路径合成 os.path.join() 在Windows上,路径中以倒斜杠作为文件夹之间的分隔符,Linux或OS X中则是正斜杠.如果想要程序正确运行于所有操作系统上,就必须要处理这 ...
- MarkDown简单语法回顾
写在前面: 本文是我的第一篇博客文章,希望与大家共同交流.分享我们热爱的技术,努力成为一名优秀的软件工程师! 进入正文 使用MarkDown记笔记已经有些时候了,编辑器是使用的sublime text ...
- 【HTB系列】Beep
出品|MS08067实验室(www.ms08067.com) 这次挑战的是 HTB 的第5台靶机:Beep,评分很高,难度中等 靶机描述 Beep 运行了大量的服务,这对正确发掘入口点有一定的挑战,由 ...
- 剑指 Offer 50. 第一个只出现一次的字符 + 哈希表 + 有序哈希表
剑指 Offer 50. 第一个只出现一次的字符 Offer_50 题目详情 方法一:使用无序哈希表 package com.walegarrett.offer; /** * @Author Wale ...
- 在ASP.NET Core中用HttpClient(一)——获取数据和内容
在本文中,我们将学习如何在ASP.NET Core中集成和使用HttpClient.在学习不同HttpClient功能的同时使用Web API的资源.如何从Web API获取数据,以及如何直接使用Ht ...
- 面试准备——计算机网络(https)
一.为什么要提出HTTPS? HTTP的缺点: 明文通信.不加密,可能被窃听. 无身份验证,可能遭遇伪装. 无法证明报文的完整性,可能被篡改. 二.HTTPS = HTTP+加密(防窃听)+认证(防伪 ...
- java 给时间增加时间得到一个新的时间(日期)
SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd") LocalDate expirationDate String exp ...