参考: https://sylvanassun.github.io/2018/03/16/2018-03-16-map_family/

Map是一种用于快速查找的数据结构,它以键值对的形式存储数据,每一个键都是唯一的,且对应着一个值,如果想要查找Map中的数据,只需要传入一个键,Map会对键进行匹配并返回键所对应的值,可以说Map其实就是一个存放键值对的集合。Map被各种编程语言广泛使用,只不过在名称上可能会有些混淆,像Python中叫做字典(Dictionary),也有些语言称其为关联数组(Associative Array),但其实它们都是一样的,都是一个存放键值对的集合。至于Java中经常用到的HashMap也是Map的一种,它被称为散列表,关于散列表的细节我会在本文中解释HashMap的源码时提及。

Java还提供了一种与Map密切相关的数据结构:Set,它是数学意义上的集合,特性如下:

  • 无序性:一个集合中,每个元素的地位都是相同的,元素之间也都是无序的。不过Java中也提供了有序的Set,这点倒是没有完全遵循。

  • 互异性:一个集合中,任何两个元素都是不相同的。

  • 确定性:给定一个集合以及其任一元素,该元素属于或者不属于该集合是必须可以确定的。

很明显,Map中的key就很符合这些特性,Set的实现其实就是在内部使用Map。例如,HashSet就定义了一个类型为HashMap的成员变量,向HashSet添加元素a,等同于向它内部的HashMap添加了一个key为a,value为一个Object对象的键值对,这个Object对象是HashSet的一个常量,它是一个虚拟值,没有什么实际含义,源码如下:

private transient HashMap<E,Object> map;
// Dummy value to associate with an Object in the backing Map
private static final Object PRESENT = new Object();
public boolean add(E e) {return map.put(e, PRESENT)==null;
}

小插曲过后,让我们接着说Map,它是JDK的一个顶级接口提供了三种集合视图(Collection Views)包含所有key的集合包含所有value的集合以及包含所有键值对的集合Map中的元素顺序与它所返回的集合视图中的元素的迭代顺序相关,也就是说,Map本身是不保证有序性的,当然也有例外,比如TreeMap就对有序性做出了保证,这主要因为它是基于红黑树实现的。

所谓的集合视图就是由集合本身提供的一种访问数据的方式,同时对视图的任何修改也会影响到集合。好比Map.keySet()返回了它包含的key的集合,如果你调用了Map.remove(key)那么keySet.contains(key)也将返回false,再比如说Arrays.asList(T)可以把一个数组封装成一个List,这样你就可以通过List的API来访问和操作这些数据,如下列示例代码:

String[] strings = {"a", "b", "c"};
List<String> list = Arrays.asList(strings);
System.out.println(list.get(0)); // "a"
strings[0] = "d";
System.out.println(list.get(0)); // "d"
list.set(0, "e");
System.out.println(strings[0]); // "e"

是不是感觉很神奇,其实Arrays.asList()只是将传入的数组与Arrays中的一个内部类ArrayList注意,它与java.util包下的ArrayList不是同一个)做了一个”绑定“,在调用get()时会直接根据下标返回数组中的元素,而调用set()时也会直接修改数组中对应下标的元素。相对于直接复制来说,集合视图的优点是内存利用率更高,假设你有一个数组,又很想使用List的API来操作它,那么你不用new一个ArrayList以拷贝数组中的元素,只需要一点额外的内存(通过Arrays.ArrayList对数组进行封装),原始数据依然是在数组中的,并不会复制成多份

Map接口规范了Map数据结构的通用API(也含有几个用于简化操作的default方法,default是JDK8的新特性,它是接口中声明的方法的默认实现,即非抽象方法)并且还在内部定义了Entry接口(键值对的实体类),在JDK中提供的所有Map数据结构都实现了Map接口,下面为Map接口的源码(代码中的注释太长了,基本都是些实现的规范,为了篇幅我就尽量省略了)。

package java.util;
import java.util.function.BiConsumer;
import java.util.function.BiFunction;
import java.util.function.Function;
import java.io.Serializable;
public interface Map<K,V> {// 查询操作/*** 返回这个Map中所包含的键值对的数量,如果大于Integer.MAX_VALUE,* 则应该返回Integer.MAX_VALUE。*/int size();/*** Map是否为空。*/Boolean isEmpty();/*** Map中是否包含key,如果是返回true,否则false。*/Boolean containsKey(Object key);/*** Map中是否包含value,如果是返回true,否则false。*/Boolean containsValue(Object value);/*** 根据key查找value,如果Map不包含该key,则返回null。*/V get(Object key);// 修改操作/*** 添加一对键值对,如果Map中已含有这个key,那么新value将覆盖掉旧value,* 并返回旧value,如果Map中之前没有这个key,那么返回null。*/V put(K key, V value);/*** 删除指定key并返回之前的value,如果Map中没有该key,则返回null。*/V remove(Object key);// 批量操作/*** 将指定Map中的所有键值对批量添加到当前Map。*/void putAll(Map<? extends K, ? extends V> m);/*** 删除Map中所有的键值对。*/void clear();// 集合视图/*** 返回包含Map中所有key的Set,对该视图的所有修改操作会对Map产生同样的影响,反之亦然。*/Set<K> keySet();/*** 返回包含Map中所有value的集合,对该视图的所有修改操作会对Map产生同样的影响,反之亦然。*/Collection<V> values();/*** 返回包含Map中所有键值对的Set,对该视图的所有修改操作会对Map产生同样的影响,反之亦然。*/Set<Map.Entry<K, V>> entrySet();/*** Entry代表一对键值对,规范了一些基本函数以及几个已实现的类函数(各种比较器)。*/interface Entry<K,V> {K getKey();V getValue();V setValue(V value);Boolean equals(Object o);int hashCode();public static <K extends Comparable<? super K>, V> Comparator<Map.Entry<K,V>> comparingByKey() {return (Comparator<Map.Entry<K, V>> & Serializable)(c1, c2) -> c1.getKey().compareTo(c2.getKey());}public static <K, V extends Comparable<? super V>> Comparator<Map.Entry<K,V>> comparingByValue() {return (Comparator<Map.Entry<K, V>> & Serializable)(c1, c2) -> c1.getValue().compareTo(c2.getValue());}public static <K, V> Comparator<Map.Entry<K, V>> comparingByKey(Comparator<? super K> cmp) {Objects.requireNonNull(cmp);return (Comparator<Map.Entry<K, V>> & Serializable)(c1, c2) -> cmp.compare(c1.getKey(), c2.getKey());}public static <K, V> Comparator<Map.Entry<K, V>> comparingByValue(Comparator<? super V> cmp) {Objects.requireNonNull(cmp);return (Comparator<Map.Entry<K, V>> & Serializable)(c1, c2) -> cmp.compare(c1.getValue(), c2.getValue());}}// 比较和hashing/*** 将指定的对象与此Map进行比较是否相等。*/Boolean equals(Object o);/*** 返回此Map的hash code。*/int hashCode();// 默认方法(非抽象方法)/*** 根据key查找value,如果该key不存在或等于null则返回defaultValue。*/default V getOrDefault(Object key, V defaultValue) {V v;return (((v = get(key)) != null) || containsKey(key)) ? v : defaultValue;}/*** 遍历Map并对每个键值对执行指定的操作(action)。* BiConsumer是一个函数接口(具有一个抽象方法的接口,用于支持Lambda),* 它代表了一个接受两个输入参数的操作,且不返回任何结果。* 至于它奇怪的名字,根据Java中的其他函数接口的命名规范,Bi应该是Binary的缩写,意思是二元的。*/default void forEach(BiConsumer<? super K, ? super V> action) {Objects.requireNonNull(action);for (Map.Entry<K, V> entry : entrySet()) {K k;V v;try {k = entry.getKey();v = entry.getValue();}catch(IllegalStateException ise) {// this usually means the entry is no longer in the map.throw new ConcurrentModificationException(ise);}action.accept(k, v);}}/*** 遍历Map,然后调用传入的函数function生成新value对旧value进行替换。* BiFunction同样是一个函数接口,它接受两个输入参数并且返回一个结果。*/default void replaceAll(BiFunction<? super K, ? super V, ? extends V> function) {Objects.requireNonNull(function);for (Map.Entry<K, V> entry : entrySet()) {K k;V v;try {k = entry.getKey();v = entry.getValue();}catch(IllegalStateException ise) {// this usually means the entry is no longer in the map.throw new ConcurrentModificationException(ise);}// ise thrown from function is not a cme.v = function.apply(k, v);try {entry.setValue(v);}catch(IllegalStateException ise) {// this usually means the entry is no longer in the map.throw new ConcurrentModificationException(ise);}}}/*** 如果指定的key不存在或者关联的value为null,则添加键值对。*/default V putIfAbsent(K key, V value) {V v = get(key);if (v == null) {v = put(key, value);}return v;}/*** 当指定key关联的value与传入的参数value相等时删除该key。*/default Boolean remove(Object key, Object value) {Object curValue = get(key);if (!Objects.equals(curValue, value) ||(curValue == null && !containsKey(key))) {return false;}remove(key);return true;}/*** 当指定key关联的value与oldValue相等时,使用newValue进行替换。*/default Boolean replace(K key, V oldValue, V newValue) {Object curValue = get(key);if (!Objects.equals(curValue, oldValue) ||(curValue == null && !containsKey(key))) {return false;}put(key, newValue);return true;}/*** 当指定key关联到某个value时进行替换。*/default V replace(K key, V value) {V curValue;if (((curValue = get(key)) != null) || containsKey(key)) {curValue = put(key, value);}return curValue;}/*** 当指定key没有关联到一个value或者value为null时,调用mappingFunction生成值并添加键值对到Map。* Function是一个函数接口,它接受一个输入参数并返回一个结果,如果mappingFunction返回的结果* 也为null,那么将不会调用put。*/default V computeIfAbsent(K key,Function<? super K, ? extends V> mappingFunction) {Objects.requireNonNull(mappingFunction);V v;if ((v = get(key)) == null) {V newValue;if ((newValue = mappingFunction.apply(key)) != null) {put(key, newValue);return newValue;}}return v;}/*** 当指定key关联到一个value并且不为null时,调用remappingFunction生成newValue,* 如果newValue不为null,那么进行替换,否则删除该key。*/default V computeIfPresent(K key,BiFunction<? super K, ? super V, ? extends V> remappingFunction) {Objects.requireNonNull(remappingFunction);V oldValue;if ((oldValue = get(key)) != null) {V newValue = remappingFunction.apply(key, oldValue);if (newValue != null) {put(key, newValue);return newValue;} else {remove(key);return null;}} else {return null;}}/*** remappingFunction根据key与其相关联的value生成newValue,* 当newValue等于null时删除该key,否则添加或者替换旧的映射。*/default V compute(K key,BiFunction<? super K, ? super V, ? extends V> remappingFunction) {Objects.requireNonNull(remappingFunction);V oldValue = get(key);V newValue = remappingFunction.apply(key, oldValue);if (newValue == null) {// delete mappingif (oldValue != null || containsKey(key)) {// something to removeremove(key);return null;} else {// nothing to do. Leave things as they were.return null;}} else {// add or replace old mappingput(key, newValue);return newValue;}}/*** 当指定key没有关联到一个value或者value为null,将它与传入的参数value* 进行关联。否则,调用remappingFunction生成newValue并进行替换。* 如果,newValue等于null,那么删除该key。*/default V merge(K key, V value,BiFunction<? super V, ? super V, ? extends V> remappingFunction) {Objects.requireNonNull(remappingFunction);Objects.requireNonNull(value);V oldValue = get(key);V newValue = (oldValue == null) ? value :remappingFunction.apply(oldValue, value);if(newValue == null) {remove(key);} else {put(key, newValue);}return newValue;}
}

需要注意一点,这些default方法都是非线程安全的任何保证线程安全的扩展类都必须重写这些方法,例如ConcurrentHashMap

下图为Map的继承关系结构图,它也是本文接下来将要分析的Map实现类的大纲,这些实现类都是比较常用的,在JDK中Map的实现类有几十个,大部分都是我们用不到的,限于篇幅原因就不一一讲解了(本文包含许多源码与对实现细节的分析,建议读者抽出一段连续的空闲时间静下心来慢慢阅读)。

Map大家族的那点事儿(一) Map相关推荐

  1. importnew:Map大家族的那点事儿

    Map大家族的那点事儿(1) :Map Map大家族的那点事儿(2) :AbstractMap Map大家族的那点事儿(3) :TreeMap Map大家族的那点事儿(4) :HashMap Map ...

  2. 安卓取map集合转换为json_android json解析成map格式

    "discount": { "3": "34", "4": "33", "5": ...

  3. java map 队列_Java:queue队列,map集合

    该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 Queue: 基本上,一个队列就是一个先入先出(FIFO)的数据结构 Queue接口与List.Set同一级别,都是继承了Collection接口.Lin ...

  4. 一个listMap里map其中的一个字段的值相同,如何判断这个字段相同,就把这个map的其他字段存入另一个map中...

    //不建议使用Map保存这些,使用实体bean更好 package com.rxlamo.zhidao; import java.util.*; public class Main {     pub ...

  5. 【Groovy】map 集合 ( map 集合定义 | 通过 getClass 函数获取 map 集合的类型 | 代码示例 )

    文章目录 一.map 集合定义 二.获取 map 集合类型 三.代码示例 一.map 集合定义 声明键值对 , 其中 键 Key 可以 不使用引号 , 可以 使用单引号 '' , 也可以 使用双引号 ...

  6. map写法 scala语言_Scala中的Map使用例子

    Map结构是一种非常常见的结构,在各种程序语言都有对应的api,由于Spark的底层语言是Scala,所以有必要来了解下Scala中的Map使用方法. (1)不可变Map 特点: api不太丰富 如果 ...

  7. java中map怎么遍历,Java中怎么遍历Map的所有的元素

    Java中怎样遍历Map的所有的元素 JDK1.4中 view plaincopy to clipboardprint? Map map = new HashMap(); Iterator it = ...

  8. Java将map置空_Java实现过滤掉map集合中key或value为空的值示例

    Java实现过滤掉map集合中key或value为空的值示例 发布时间:2020-09-16 23:26:14 来源:脚本之家 阅读:147 作者:May的博客 本文实例讲述了Java实现过滤掉map ...

  9. c++ map初始化同时赋值_Golang入门教程——map篇

    点击上方蓝字,和我一起学技术. 今天是golang专题的第7篇文章,我们来聊聊golang当中map的用法. map这个数据结构我们经常使用,存储的是key-value的键值对.在C++/java当中 ...

最新文章

  1. Java学习总结:37(比较器)
  2. 计算机应用基础课程是过程化考试吗,基于能力的计算机应用基础课程过程化考核标准构建与实施...
  3. XML Schema用法
  4. 【转】系统管理类DOS命令汇总
  5. svd medium_我们刚刚放弃了Medium博客。 您可能也应该这样做。
  6. Google 被祭天了!
  7. Qt编译时报堆空间不足
  8. Java项目中常见的文件夹名称
  9. matlab 模糊提取,[转载]Matlab 的fspecial函数用法 图像模糊、提取边缘
  10. mysql update语句的用法详解
  11. 上传下载文件实例(vsftp服务器+nginx)
  12. win10系统小米妙享中心,在手机可搜索到电脑,与之跨屏协作
  13. 工作十年之感悟 -- 兼谈生活与人生
  14. C语言修改dos窗口的大小
  15. 什么是TDK?什么是网站的TDK?扫(myself的)盲
  16. 图片浏览器功能的实现(一)——图片放大与缩小功能实现
  17. 当在浏览器地址栏输入一个URL后回车,将会发生的事情?
  18. fatal: unable to access ‘https://github.com/xxx/123.git/‘: Failed connect to github.com:443 解决方案
  19. php一句话-D盾绕过小技巧
  20. asp.nett网站发布过程

热门文章

  1. Python运用蒙特卡洛算法模拟植物生长
  2. 传智播客免费IT学习资源站-视频库隆重上线
  3. 树莓派如何终端手动配网
  4. 未来5G物联网的安全形态,看英飞凌eSIM
  5. SAP 银企直连 通过 Http Get 方式下载交易明细文件
  6. 基于深度学习的图标型验证码识别系统
  7. matlab图像导数求积分_第二讲matlab求微分方程导数积分
  8. Python多继承的C3算法
  9. 4. 写第一篇博客,最好的时间是今日,今时,今刻
  10. 查看每个cpu核使用情况