Map中经常被忽略但又非常好用的方法
1. 简介
map是我们日常开发中常会的集合类之一, 但是我们除了常用的get和put之外,其他的方法好像很少会用到,接下来我们就介绍一下几个经常被忽略但又很好用的方法.
2. Quick Start
2.1 数据准备
创建一个map对象, 并声明几个用于测试的user对象
Map<Integer, User> hashMap = Maps.newHashMap();
User zhangsan = new User(1, "张三");
User lisi = new User(2, "李四");
User zhangtieniu = new User(3, "张铁牛");
2.2 重温put
// hashmap put (添加/更新元素)
@Test
public void put() {
User test = hashMap.put(null, null);
User user = hashMap.put(1, null);
User user1 = hashMap.put(1, zhangsan);
User user2 = hashMap.put(1, lisi);
User user3 = hashMap.put(null, zhangsan);
User user4 = hashMap.put(null, lisi);
User user5 = hashMap.get(null);
log.info("map: {}", hashMap);
log.info("user: {}, user1: {}, user2: {}, user3: {}, user4: {}, user5: {}", user, user1, user2, user3, user4, user5);
//map: {null=User(id=2, name=李四), 1=User(id=2, name=李四)}
//user: null, user1: null, user2: User(id=1, name=张三), user3: null, user4: User(id=1, name=张三), user5: User(id=2, name=李四)
}
- key和value可以为null (hashmap 和 linkedhashmap)
- 使用null可以正常的覆盖和获取元素
- put可以直接新增or覆盖已有的元素
- put方法返回对应key的oldValue,如果没有oldValue则返回null
2.3 getOrDefault
// getOrDefault(Object key, V defaultValue) (获取/返回默认值)
@Test
public void getOrDefault() {
hashMap.put(1, zhangsan);
hashMap.put(2, null);
final User user1 = hashMap.get(1);
final User user2 = hashMap.getOrDefault(2, lisi);
final User user3 = hashMap.getOrDefault(3, zhangtieniu);
log.info("map: {}", hashMap);
//map: {1=User(id=1, name=张三), 2=null}
log.info("user1: {}, user2: {}, user3: {}", user1, user2, user3);
//user1: User(id=1, name=张三), user2: null, user3: User(id=3, name=张铁牛)
}
当map中没有对应的key时, 返回对应的defaultValue
注意: 如果map中存在对应的key, 但是对应的
value == null时, 返回的是null, 而不是defaultValue
源码如下:
default V getOrDefault(Object key, V defaultValue) {
V v;
return (((v = get(key)) != null) || containsKey(key))
? v
: defaultValue;
}
2.4 putIfAbsent
// putIfAbsent(K key, V value) (如果不存在则添加)
@Test
public void putIfAbsent() {
hashMap.put(1, null);
User user1 = hashMap.putIfAbsent(1, zhangsan);
User user2 = hashMap.putIfAbsent(2, lisi);
User user3 = hashMap.putIfAbsent(2, zhangtieniu);
log.info("map: {}", hashMap);
log.info("user1: {}, user2: {}, user3: {}", user1, user2, user3);
//map: {1=User(id=1, name=张三), 2=User(id=2, name=李四)}
//user1: null, user2: null, user3: User(id=2, name=李四)
}
- 如果指定的key对应的value不为null时(
oldValue != null) : 不覆盖 & 返回oldValue - 当指定key的value不存在时(
oldValue == null) : 添加元素 & 返回oldValue
可以理解为 当指定key的value不存在时, 才去put, 否则不添加
源码如下:
default V putIfAbsent(K key, V value) {
V v = get(key);
if (v == null) {
v = put(key, value);
}
return v;
}
2.5 compute
// compute(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction) (计算)
@Test
public void compute() {
hashMap.put(1, zhangsan);
User user1 = hashMap.compute(1, (k, oldValue) -> lisi);
log.info("map: {}, user1: {}", hashMap, user1);
//map: {1=User(id=2, name=李四)}, user1: User(id=2, name=李四)
User user2 = hashMap.compute(1, (k, oldValue) -> null);
log.info("map: {}, user2: {}", hashMap, user2);
//map: {}, user2: null
}
remappingFunction返回值 != null: 覆盖oldValue & 返回newValueremappingFunction返回值 == null: 删除对应元素 & 返回null
可以理解为 使用remappingFunction的返回值覆盖对应key的旧值, 当remappingFunction返回值为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;
}
}
2.6 computeIfAbsent
// computeIfAbsent(K key, Function<? super K, ? extends V> mappingFunction) (不存在则计算)
@Test
public void computeIfAbsent() {
User user = hashMap.computeIfAbsent(1, k -> zhangsan);
User user1 = hashMap.computeIfAbsent(1, k -> lisi);
User user2 = hashMap.computeIfAbsent(2, k -> null);
log.info("map: {}, user:{}, user1: {}, user2:{}", hashMap, user, user1, user2);
//map: {1=User(id=1, name=张三)}, user:User(id=1, name=张三), user1: User(id=1, name=张三), user2:null
}
oldValue != null: 不覆盖 & 返回oldValueoldValue == null && mappingFunction返回值 != null: 添加元素 & 返回newValueoldValue == null && mappingFunction返回值 == null: 不覆盖 & 返回oldValue
可以理解为 当指定key的value不存在时, 才使用mappingFunction的返回值覆盖对应key的旧值, 如果key对应value存在或者mappingFunction的返回值为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;
}
2.7 computeIfPresent
// computeIfPresent(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction) (存在则计算)
@Test
public void computeIfPresent() {
hashMap.put(1, zhangsan);
User user1 = hashMap.computeIfPresent(1, (k,oldValue) -> lisi);
User user2 = hashMap.computeIfPresent(3, (k,oldValue) -> zhangtieniu);
log.info("map: {}, user1: {}, user2: {}", hashMap, user1, user2);
//map: {1=User(id=2, name=李四)}, user1: User(id=2, name=李四), user2: null
User user3 = hashMap.computeIfPresent(1, (k,oldValue) -> null);
log.info("map: {}, user3:{}", hashMap, user3);
//map: {}, user3:null
}
oldValue == null: 不覆盖&返回nulloldValue != null && remappingFunction返回值 == null: 移除元素&返回nulloldValue != null && remappingFunction返回值 != null: 覆盖元素&返回newValue
可以理解为 当key对应的value存在时, 才使用remappingFunction的返回值覆盖对应key的旧值, 如果key对应的value不存在或者remappingFunction的返回值为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;
}
}
2.8 replace
// replace(K key, V value)
// replace(K key, V oldValue, V newValue) (替换)
@Test
public void replace() {
hashMap.put(1, zhangsan);
hashMap.put(2, lisi);
hashMap.replace(1, zhangtieniu);
hashMap.replace(2, null);
hashMap.replace(3, zhangtieniu);
hashMap.replace(2, null, zhangtieniu);
log.info("map: {}", hashMap);
//map: {1=User(id=3, name=张铁牛), 2=User(id=3, name=张铁牛)}
}
- 替换指定key的value值
- 可以将对应的value设置为null
- 对应的key不存在时不会添加新元素
replace(K key, V oldValue, V newValue)方法多了一层判断, 当key对应的value与oldValue相等时, 才会替换newValue
可以理解为 替换指定key的value值
源码如下:
default V replace(K key, V value) {
V curValue;
if (((curValue = get(key)) != null) || containsKey(key)) {
curValue = put(key, value);
}
return curValue;
}
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;
}
2.9 merge
// merge(K key, V value, BiFunction<? super V, ? super V, ? extends V> remappingFunction) (融合)
@Test
public void merge() {
hashMap.put(1, zhangsan);
User user1 = hashMap.merge(1, lisi, (oldValue, defaultValue) -> zhangtieniu);
User user2 = hashMap.merge(2, lisi, (oldValue, defaultValue) -> zhangtieniu);
log.info("map: {}, user1: {}, user2: {}", hashMap, user1, user2);
//map: {1=User(id=3, name=张铁牛), 2=User(id=2, name=李四)}, user1: User(id=3, name=张铁牛), user2: User(id=2, name=李四)
}
oldValue == null: 使用传进来的 value 作为newValue,oldValue != null: 使用remappingFunction的返回值作为newValuenewValue == null: 移除元素 & 返回newValuenewValue != null: 覆盖元素 & 返回newValue
可以理解为 融合三个值 分别为:
- key对应的value(oldValue)
- merge方法的第二个参数value (可以理解为oldValue的defaultValue)
- merge方法的第三个参数remappingFunction方法的返回值
融合逻辑为: 如果key对应的value不存在时, 使用merge方法的第二个参数value作为newValue, 如果key对应的value存在时,使用remappingFunction的返回值作为newValue, 如果newValue不为null则覆盖元素, 为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;
}
}
2.10 foreach
// foreach java8 新增
@Test
public void foreach() {
hashMap.put(1, zhangsan);
hashMap.put(2, lisi);
hashMap.forEach((key, value) -> log.info("key: {}, value: {}", key, value));
//key: 1, value: User(id=1, name=张三)
//key: 2, value: User(id=2, name=李四)
}
3. 总结
| 方法名称 | 方法参数 | 方法描述 | 方法特点 |
|---|---|---|---|
| put | key, value | 添加元素 | hashmap/linkedhashmap: key, value 都可以为null |
| getOrDefault | key, defaultValue | 获取元素 | 当map中没有对应的key时, 返回defaultValue |
| putIfAbsent | key, value | 当不存在时添加元素 | 这里不存在指的是: key对应的旧值为null |
| compute | key, BiFunction<key, oldValue, newValue> remappingFunction | 重新计算key对应的value | 使用remappingFunction的返回值替换key的旧值 |
| computeIfAbsent | key, Function<key, value> mappingFunction | 当不存在时计算 | 这里不存在指的是: key对应的旧值为null, 与putIfAbsent方法逻辑类似 |
| computeIfPresent | key, BiFunction<key, oldValue, newValue> remappingFunction | 当存在时计算 | 这里存在指的是: key对应的旧值!=null |
| replace | key, value | 替换元素 | 替换指定key的value, 不会添加元素 |
| merge | key, value, BiFunction<oldValue, value, newValue> remappingFunction | 融合 | 融合key的旧值, 默认值, remappingFunction的返回值作为新值 |
| foreach | BiConsumer<key, value> | 遍历 | java8新加的遍历方式 |
Map中经常被忽略但又非常好用的方法的更多相关文章
- Map java中的map 如何修改Map中的对应元素
Map java中的map 如何修改Map中的对应元素 Map以按键/数值对的形式存储数据,和数组非常相似,在数组中存在的索引,它们本身也是对象. Map的接口 Map ...
- 自定义注解+反射提取对象到map中
一.问题:有时候我们与第三方接口对接传参时,需要将对象里的字段和值以map形式传给别人,此时可以借助其他的工具类,但是我个人用起来不太灵活,还会把多余的字段传给别人,因此我们自己动手搞一套 二.思路: ...
- jsp页面使用el 按key获取map中的对应值
jsp页面使用el 按key获取map中的对应值 转自:<jsp页面使用el 按key获取map中的对应值>地址:http://blog.csdn.net/baple/article/de ...
- 3. 如何封装查询条件与查询结果到map中
public Map<String, Object> queryOrderStatus(String orderNo) { // 查询到的结果与查询的条件一一对应,封装到map中! Str ...
- Java根据条件删除Map中元素
今天在写程序过程中,需要根据判断条件删除一个Map中的相应数据,我自然而然想到可以通过调用Map中的remove(Object key)函数进行删除:代码如下: public Map<Doubl ...
- Map中如何把没有定义操作符<的类作为key
Map中如何把没有定义操作符<的类作为key 其实,为了实现快速查找,map内部本身就是按序存储的(比如红黑树).在我们插入<key, value>键值对时,就会按照key的大小顺序 ...
- 对象属性封装到map中
import java.beans.PropertyDescriptor; import java.lang.reflect.Method; import java.lang.reflect.Modi ...
- map中的erase成员函数用法
转载于 http://www.cnblogs.com/graphics/archive/2010/07/05/1771110.html http://hi.baidu.com/sdkinger/it ...
- 修改Map中确定key对应的value问题
今天在码代码的时候出现一个没有预料的问题: 先看下面的代码: public static void main(String[] args) { String[] files=new String[]{ ...
- Ibatis中sqlmap参数map中还需要套list的情况如何写?
原始需求: 有若干个参数,需要作为ibatis拼装sql的参数传入,但是有个参数的值比较特殊,是若干种枚举值.具体到这个case,就是有有限个namespace.我每次需要通过传入多个namespac ...
随机推荐
- Junit5
JUnit5 安卓build.gradle https://github.com/mannodermaus/android-junit5 Unit 3 或 JUnit4 的向后兼容性 JUnit4 已 ...
- C语言刷题小知识点
力扣返回二维数组 int **spiralMatrix(int m, int n, struct ListNode *head, int *returnSize, int **returnColumn ...
- Sql高级
sql高级 1. 索引与视图 常见的数据结构 栈:先进后出 队列:先进先出 数组:查询快,根据下标查询 链表:分为双链表与单链表.单链表指向下一个数据的存储位置:双链表指向前一个与下一个数据的存储位置 ...
- git安装-Tortoise git 安装汉化教程
1.首先下载 去官网下载 如果下载比较慢的,链接自取 https://pan.quark.cn/s/fcb9d0b39c7f 2. 安装git 3. 安装git图形化工具Tortoise git 4. ...
- Java创建数组、赋值的四种方式,声明+创建+初始化 详解
@ 目录 一.创建数组的四种方式 二.详解 三.数组存储的弊端 一.创建数组的四种方式 以int数据类型为例 @Test public void testNewArray() { //创建数组 //法 ...
- 20240719 CVTE 笔试
岗位:嵌入式软件开发工程师(Linux方向) 题型:20 道不定项选择题,2 道编程题 1.不定项选择题 1.1 如下哪个命令可以帮助你知道 shell 的用法 (D) more help pwd m ...
- springboot将文件处理成压缩文件
前言 在工作我们经常会出现有多个文件,为了节省资源会将多个文件放在一起进行压缩处理:为了让大家进一步了解我先将springboot处理的方法总结如下,有不到之处敬请大家批评指正! 一.文件准备: ht ...
- P4119 Ynoi2018 未来日记
P4119 Ynoi2018 未来日记 lxl 出的题好 duliu 啊. 感谢来自 fr200110217102 的博客 题解 P4119 [Ynoi2018未来日记]. 下标分块+值域分块+并查集 ...
- Vulhub Nginx漏洞复现
目录 前言 文件名逻辑漏洞(CVE-2013-4547) 配置不当导致解析漏洞 配置错误导致漏洞 CRLF注入漏洞 目录穿越漏洞 前言 Nginx是一款广泛使用的Web服务器和反向代理服务器,尽管它以 ...
- python 快速比较大文件的元素异同之处
0x00 问题 0x01 解决方法 0x02 list最多可以存放多少条数据呢? 0x03 集合set的操作 0x00 问题 假如,在有两个大文件分别存储了大量的数据,数据其实很简单就是一堆字符串,每 ...