第一次写笔记就从map开始吧,如上图所示,绿色的是interface,黄色的是abstract class,蓝色的是class,可以看出所有和图相关的接口,抽象类和类的起源都是interface map<K,V>。现在看一下当前Map接口中实现了什么,里面包含了一个内部接口interface Entry<K,V>,声明了14个方法,同时还定义了11个方法(在jdk1.8中,在interface中可以使用default关键字来定义完整的方法)

由于声明的方法都是原来版本jdk已经存在的,比较简单,所以我们从声明的方法开始介绍

 public interface Map<K, V> {

     /**
* 方法返回键值对个数,当键值对个数大于Integer.MAX_VALUE
* 时返回Integer.MAX_VALUE
*/
int size(); /**
* 当Map中没有键值对的时候,返回true
*/
boolean isEmpty(); /**
* 当Map中的键包含key的时候返回true,相当于执行了如下语句
* key==null ? k==null : key.equals(k)
* 此处应该注意Map中的键是可以为null的,但是只可以有一个
* 需要注意的是本方法会有两个异常
* ClassCastException当传入的key和本Map中设置的key类型不相容的时候
* NullPointerException当传入的key==null,并且Map中的键没有null的时候
*/
boolean containsKey(Object key); /**
* 当有多于一个key的value等于传入的value时会返回true
* 相当于执行value==null ? v==null : value.equals(v)
* 可见value也是可以为null的
* 需要注意的是本方法会抛出两个异常
* ClassCastException当传入的值和Map中的值不相容的时候
* NullPointerException当value中没有一个值为null的时候
*/
boolean containsValue(Object value); /**
* 如果当前的key存在于Map中,会返回对应的value,否则会返回null
* 需要注意的是本方法会有两个异常
* ClassCastException当传入的key和本Map中设置的key类型不相容的时候
* NullPointerException当传入的key==null,并且Map中的键没有null的时候
*
*/
V get(Object key); /**
* 当key不存在是会放入新的键值对,当key存在时,会更新key对应的value值
* 本方法会产生四个异常
* UnsupportedOperationException当Map不支持put操作的时候
* ClassCastException当key或者value不相容的时候
* NullPointerException当Map不允许key或者value为null的时候
* IllegalArgumentException如果指定键或值的某些属性阻止将其存储在此映射中
*/
V put(K key, V value); /**
* 当Map中有这个key的时候,会返回相应的value,否则会返回null
* 本方法会抛出三个异常
* UnsupportedOperationException当Map不支持此操作时
* ClassCastException当key的类型不相容时
* NullPointerException当key是null并且这个Map不允许key为null时
*/
V remove(Object key); /**
* 将一个Map的所有键值对放入本Map,对于每一个键值对都调用的是put(K key, V value)方法
* 本方法会产生四个异常
* UnsupportedOperationException当Map不支持put操作的时候
* ClassCastException当key或者value不相容的时候
* NullPointerException当Map不允许key或者value为null的时候
* IllegalArgumentException如果指定键或值的某些属性阻止将其存储在此映射中
*/
void putAll(Map<? extends K, ? extends V> m); /**
* 从Map中移除所有的元素
* 会抛出一个异常
* UnsupportedOperationException当Map不支持本操作的时候
*/
void clear(); /**
* 会以集合的形式返回key,当修改Map的key时会反应在Set中,反之亦然
*/
Set<K> keySet(); /**
* 返回一个Collection集合对于值来说,因为值可以有重复的,所以才选用了Collection
* 同样的对Map修改时,也会反应到Collection上
*/
Collection<V> values(); /**
* 返回一个键值对的集合Entry<K, V>
*/
Set<Map.Entry<K, V>> entrySet(); /**
* 本方法是用来比较两个Map是否相同的
*/
boolean equals(Object o); /**
* 本方法是用于返回一个Map的Hash值,本方法要求对Map中的每一个Entry<K,V>进行运算的
* 这样保证了当m1.equals(m2)时m1.hashCode()==m2.hashCode()
*/
int hashCode(); }

接下来我们来看一下内部的接口interface Entry<K,V>的内容

 interface Entry<K,V> {
/**
* 返回Entry的key
* IllegalStateException 并不要求实现类必须抛出这个异常,在访问Map中的Entry元素时可能用到,即Entry已经被移除时
*/
K getKey(); /**
* 返回Entry的value
* IllegalStateException 并不要求实现类必须抛出这个异常,在访问Map中的Entry元素时可能用到,即Entry已经被移除时
*/
V getValue(); /**
* 重新设置Entry的value,但是当映射关系被移除时,该方法的行为还未被定义
* 抛出五个异常
* UnsupportedOperationException如果Map不支持put操作
* ClassCastException当与指定类型不相容的时候
* NullPointerException当Map不允许出现null而此时设置的值为null的时候
* IllegalArgumentException如果当前值的属性不允许它存储在图中
* IllegalStateException并不要求实现类必须抛出这个异常,在访问Map中的Entry元素时可能用到,即Entry已经被移除时
*/
V setValue(V value); /**
* 当比较对象也是一个Entry并且表达同样的映射关系的时候返回true
*/
boolean equals(Object o); /**
* 本方法实现的要求是当e1.equals(e2)时,e1.hashCode()==e2.hashCode()
*/
int hashCode(); //从以下开始都是jdk1.8新增的方法,对于jdk1.8的新特性就不再这里分析了,会单写一篇的
/**
* 本方法会返回一个比较器,用来实现对Entry的key进行比较
*/
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());
} /**
* 本方法返回一个比较器,用来实现对Entry的value进行比较,这里表示返回值同时满足两个接口
*/
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());
} /**
* 这个方法和上两个方法不同之处在于上两个要求K或者V实现Comparable接口,使得本身可以比较
* 而这个方法中并没有对K,V有什么要求,而是需要一个已经实现Comparator的接口
*/
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());
} }

这里简单解释一下上述代码中jdk1.8的新特性以便使代码容易理解一些,Comparator<Map.Entry<K, V>> & Serializable这里并不是表达与操作,而是返回值同时实现了这两个接口,(c1, c2) -> cmp.compare(c1.getValue(), c2.getValue())是lambda表达式实现了一个函数式接口Comparator的compare方法

最后一部分是Map在jdk1.8中新增的default方法,在jdk1.8中,可以在interface中实现方法了,但是得用default关键字修饰

     /**
* 本方法是在普通get方法上增加了一个默认值,防止了得到的value为null的情况
* 抛出两个异常
* ClassCastException当key和Map中要求的key不兼容时
* NullPointerException当Map不支持null为key时
*/
default V getOrDefault(Object key, V defaultValue) {
V v;
return (((v = get(key)) != null) || containsKey(key))
? v
: defaultValue;
} /**
* 对Map中的每一个Entry进行一次action包含的操作
* 抛出两个异常
* NullPointerException当action为null时
* ConcurrentModificationException在迭代过程中有元素被移除
*/
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中的每个Entry都用function进行操作,返回得到的结果重新设置value
*/
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的时候,设置新的value值
*/
default V putIfAbsent(K key, V value) {
V v = get(key);
if (v == null) {
v = put(key, value);
} return v;
} /**
* 当key和value与当前map中有完全相同的,就删除该元素,返回true,否则返回false,不做任何操作
*/
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;
} /**
* 当Map中存在key->oldValue时,替换为key->newValue,返回true,否则返回false,不做任何操作
*/
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;
} /**
* 用新的value替换老的value,并返回老的value,如果key不存在,返回null
*/
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,然后对key进行重新计算,得到新的value,返回新的value。否则返回null
*/
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;
} /**
* 如果Map中存在key,并且对应的value!=null的时候,用传入的两个参数的函数进行重新计算,得到新的value,如果新的value==null
* 就删除这个键值对,返回null,否则覆盖旧值,返回新的value,如果key不存在,也返回null
*/
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;
}
} /**
* 通过原来的键值对,计算出一个新的值,如果新的值不为空,重新设置键值对,并返回新的value,否则删除原来的键值对,返回null
*/
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 mapping
if (oldValue != null || containsKey(key)) {
// something to remove
remove(key);
return null;
} else {
// nothing to do. Leave things as they were.
return null;
}
} else {
// add or replace old mapping
put(key, newValue);
return newValue;
}
} /**
* 将旧的value和新的value进行混合计算,得到另一个value,如果这个value不为空,就和原来的key形成映射关系,返回新的value,
* 否则移除原来的键值对,返回null
*/
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;
}

这些方法中出现了比较陌生的BiConsumer,BiFunction和Function这三个接口,都是在java.util.function包下的新接口,可以把他们看做一个函数去理解就好了。BiConsumer

是接受两个参数,没有计算的返回值。BiFunction是接受两个参数返回一个参数。Function是接受一个参数返回一个参数。

随笔1 interface Map<K,V>的更多相关文章

  1. JDK源码(1.7) -- java.util.Map<K,V>

     java.util.Map<K,V> 源码分析 --------------------------------------------------------------------- ...

  2. Map<k,v>接口

    https://docs.oracle.com/javase/8/docs/api/java/util/Map.html public interface Map<K,V> K—key,V ...

  3. cocos基础教程(5)数据结构介绍之cocos2d::Map<K,V>

    1.概述 cocos2d::Map<K,V> 是一个内部使用了 std::unordered_map的关联容器模版. std::unordered_map 是一个存储了由key-value ...

  4. 关于jsp利用EL和struts2标签来遍历ValueStack的东东 ------> List<Map<K,V>> 以及 Map<K,<List<xxx>>> 的结构遍历

    //第一种结构Map<K,<List<xxx>>> <body> <% //显示map<String,List<Object>& ...

  5. Cocos2d-x之Map<K, V>

    |   版权声明:本文为博主原创文章,未经博主允许不得转载. Map<K, V>是Cocos2d-x 3.0x中推出的字典容器,它也能容纳Ref类型.Map<K,V>是模仿C+ ...

  6. Mybatis返回List<Map<K,V>>

    最终映射的字段名 会被作为 hashMap 的 key , <!-- TODO 测试返回 HashMap--> <resultMap id="testResultMap&q ...

  7. Map<K, V> 中k,v如果为null就转换

    Set<String> set = map.keySet(); if(set != null && !set.isEmpty()) { for(String key : s ...

  8. Java------遍历Map<k,v>的方法

    1. public class MapAction extends ActionSupport{ private Map<String, User> map = new HashMap&l ...

  9. Cocos2d-x3.0模版容器具体解释之二:cocos2d::Map&lt;K,V&gt;

    1.概述: 版本号: v3.0 beta 语言: C++ 定义在 "COCOS2DX_ROOT/cocos/base" 路径下的 "CCMap.h" 的头文件里 ...

随机推荐

  1. [CSP-S模拟测试]:连连看(图论+容斥)

    题目传送门(内部题74) 输入格式 输入文件$link.in$ 第一行三个整数$n,m,k$,之间用空格隔开,$n,m$表示地图行数和列数,$k$表示每个方块周围相邻的位置(至多有$4$个,至少有$2 ...

  2. Vertical Center TextView . 竖直居中的UITextView

    @interface VerticalCenterTextView : UITextView @end @implementation VerticalCenterTextView - (void) ...

  3. 5个用/不用GraphQL的理由

    我在如何使用Gatsby建立博客 / How to build a blog with Gatsby这篇文章中提过GraphQL在Gatsby中的应用.总的来讲,它是一个新潮的技术,在适宜的使用场景威 ...

  4. LNMP 搭建 wordpress 站点 安装及配置过程

    0x00 环境 阿里云ECS云服务器 CPU:1核 内存:4G 操作系统:Centos 系统盘:100G 0x01 安装及配置 主要使用 nginx . php 和 mysql 注意:如果下面的设置不 ...

  5. django小知识

    def __str__: return self.name 在显示的时候,将原来显示的额object对象,显示成这个类的名字

  6. How to use Nlog for ASP.NET Core with csproj

    1. Add dependency in csproj manually or using NuGet Install the latest: NLog.Web.AspNetCore 4.5+ Upd ...

  7. js提交图片转换为base64

    $("#picAjax").change(function () { var strs = ""; var file = $("#picAjax&qu ...

  8. 知识点C++

    比较2个字符串的大小…… s1=s2,strcmp(s1,s2) == ; s1>s2, strcmp(s1,s2) == ; s1<s2, strcmp(s1,s2) == -; str ...

  9. mysql字符串拆分实现split功能

    转自:https://blog.csdn.net/pjymyself/article/details/81668157有分隔符的字符串拆分题目要求数据库中 num字段值为: 实现的效果:需要将一行数据 ...

  10. JAVA总结--泛型

    泛型 :程序设计语言的一种特性:将类型参数化: 特征:凡是涉及到强制类型转化的地方,使用泛型均会编译出现问题:泛型仅仅在编译时进行校验,使用泛型的对象,其本质的类型依然不变: ps:不存在泛型数组 一 ...