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 ...
随机推荐
- cmd杀死占用端口号的Java进程
下面列出两种杀死进程的方法: 1.根据jps查询.2.根据端口号查询进程. 最后根据进程id杀死进程(注意:进程id不等同于端口号) 根据jps查进程 jps命令,列出Java进程列表 根据进程id杀 ...
- Mysql数据类型面试题15连问
整数类型的 UNSIGNED 属性有什么用? MySQL 中的整数类型可以使用可选的 UNSIGNED 属性来表示不允许负值的无符号整数.使用 UNSIGNED 属性可以将正整数的上限提高一倍,因为它 ...
- Redis未授权漏洞复现
目录 Redis 漏洞的产生条件及利用 Redis环境搭建 漏洞复现 利用Redis写入Webshell 利用Redis写入SSH公钥 利用Redis写入计划任务 Redis安全防护 Redis re ...
- apache+jk+tomcat集群+session同步
apache2+tomcat5.5集群+session同步 作者:刘宇 liuyu.blog.51cto.com msn群:mgroup49073@hotmail.com (linuxtone) ...
- Java深度历险(六)——Java注解——(七)——Java反射与动态代理
在开发Java程序,尤其是Java EE应用的时候,总是免不了与各种配置文件打交道.以Java EE中典型的S(pring)S(truts)H(ibernate)架构来说,Spring.Struts和 ...
- 使用<a-select>时,placeholder不起作用
当绑定v-model的值之后,placeholder设置的值不起作用,此时需要把v-model绑定的值设置为undefined就可以了
- Blazor 组件库 BootstrapBlazor 中Modal组件介绍
组件说明 Model组件是一个模态框组件,可以弹出一个对话框,适合需要定制性更大的场景. 它的样子如下: 其html代码为: <div class="modal-content&quo ...
- S2P销讯通·CRM-移动的客户关系精细化管理
S2P销讯通·CRM是一款专为医药企业设计的移动客户关系管理软件.该软件安装在手机上,集主数据管理.辖区指标管理.客户管理.SFE管理.OTC动销管理.精细化招商管理.市场活动管理以及流向采集清洗与统 ...
- 使用Docker快速部署一个Net项目
前言 Docker 可以让开发者打包他们的应用以及依赖包到一个轻量级.可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现虚拟化. 优点 Web 应用的自动化打包和发布. 自动化测试和 ...
- vue前端代码npm install报错的解决方法
npm install,报错: npm WARN tarball tarball data for has-bigints@https://registry.npmmirror.com/has-big ...