登录  /  注册
首页 > Java > java教程 > 正文

Map.merge()的详细介绍(附代码)

不言
发布: 2019-03-22 17:04:09
转载
5787人浏览过

本篇文章给大家带来的内容是关于map.merge()的详细介绍(附代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

今天介绍Map的merge方法,让我们来看看它的强大之处。

在JDK的API中,这样的一个方法它是很特别的,它很新颖,它是值得我们花时间去了解的,同时也推荐你可以运用到实际的项目代码中,对你们应该帮助很大。Map.merge())。这可能是Map中最通用的操作。但它也相当模糊,几乎很少人会去使用它。

背景介绍

merge() 可以解释如下:它将新的值赋值给到key中(如果不存在)或更新具有给定值的现有key(UPSERT)。让我们从最基本的例子开始:计算唯一的单词出现次数。在java8之前的时候,代码非常混乱,实际的实现其实已经失去了本质层面的设计意义。

var map = new HashMap<string>();
words.forEach(word -&gt; {
    var prev = map.get(word);
    if (prev == null) {
        map.put(word, 1);
    } else {
        map.put(word, prev + 1);
    }
});</string>
登录后复制

按照上述代码的逻辑,假设给定一个输入集合,输出的结果如下;

var words = List.of("Foo", "Bar", "Foo", "Buzz", "Foo", "Buzz", "Fizz", "Fizz");
//...
{Bar=1, Fizz=2, Foo=3, Buzz=2}
登录后复制

改进V1

现在让我们来重构它,主要去掉它的一些判断逻辑;

words.forEach(word -&gt; {
    map.putIfAbsent(word, 0);
    map.put(word, map.get(word) + 1);
});
登录后复制

这样的改进,是可以满足我们的重构要求。putIfAbsent()的具体用法就不过多描述。putIfAbsent那一行代码是一定需要的,否则,后面的逻辑也就会报错。而在下面代码中,又出现了put、get这一点会很奇怪,让我们再继续的进行改进设计。

改进V2

words.forEach(word -&gt; {
    map.putIfAbsent(word, 0);
    map.computeIfPresent(word, (w, prev) -&gt; prev + 1);
});
登录后复制

computeIfPresent是仅当 word中的的key存在的时候才调用给定的转换。否则它什么都不处理。我们通过将key初始化为零来确保key存在,因此增量始终有效。这样的实现是不是已经足够完美?未必,还有其他的思路可以减少额外的初始化。

words.forEach(word -&gt;
        map.compute(word, (w, prev) -&gt; prev != null ? prev + 1 : 1)
);
登录后复制

compute ()就像是computeIfPresent(),但无论给定key的存在与否如何都会调用它。如果key的值不存在,则prev参数为null。将简单移动if 到隐藏在lambda中的三元表达式也远远没有达到最佳的表现。在我向你展示最终版本之前,让我们看一下稍微简化的默认实现Map.merge()源码分析。

改进V3

merge()源码
default V merge(K key, V value, BiFunction<v> remappingFunction) {
    V oldValue = get(key);
    V newValue = (oldValue == null) ? value :
               remappingFunction.apply(oldValue, value);
    if (newValue == null) {
        remove(key);
    } else {
        put(key, newValue);
    }
    return newValue;
}</v>
登录后复制

代码片段胜过千言万语。 阅读源码总是能够发现新大陆,merge() 适用于两种情况。如果给定的key不存在,它就变成了put(key, value)。但是,如果key已经存在一些值,我们  remappingFunction 可以选择合并的方式。这个功能是完美契机上面的场景:

  • 只需返回新值即可覆盖旧值: (old, new) -&gt; new
  • 只需返回旧值即可保留旧值: (old, new) -&gt; old
  • 以某种方式合并两者,例如: (old, new) -&gt; old + new
  • 甚至删除旧值: (old, new) -&gt; null

如你所见,它  merge() 是非常通用的。那么,我们的问题该如何使用merge()呢?代码如下:

words.forEach(word -&gt;
        map.merge(word, 1, (prev, one) -&gt; prev + one)
);
登录后复制

你可以按照如下思路理解:如果没有key,那么初始化的value等于1;否则,将1添加到现有值。代码中的 one 是一个常量,因为我们的场景中,默认一直是加1,具体变化可以随意切换。

场景

想象一下,merge()真的那么好用吗?它的场景可以有什么?

举一个例子。你有一个帐户操作类

class Operation {
    private final String accNo;
    private final BigDecimal amount;
}
登录后复制

以及针对不同帐户的一系列操作:

operations = List.of(
    new Operation("123", new BigDecimal("10")),
    new Operation("456", new BigDecimal("1200")),
    new Operation("123", new BigDecimal("-4")),
    new Operation("123", new BigDecimal("8")),
    new Operation("456", new BigDecimal("800")),
    new Operation("456", new BigDecimal("-1500")),
    new Operation("123", new BigDecimal("2")),
    new Operation("123", new BigDecimal("-6.5")),
    new Operation("456", new BigDecimal("-600"))
);
登录后复制

我们希望为每个帐户计算余额(总运营金额)。假如不用merge(),就变得非常麻烦了:

Map balances = new HashMap<string>();
operations.forEach(op -&gt; {
    var key = op.getAccNo();
    balances.putIfAbsent(key, BigDecimal.ZERO);
    balances.computeIfPresent(key, (accNo, prev) -&gt; prev.add(op.getAmount()));
});</string>
登录后复制

使用merge之后的代码

operations.forEach(op -&gt;
        balances.merge(op.getAccNo(), op.getAmount(), 
                (soFar, amount) -&gt; soFar.add(amount))
);
登录后复制

再进行优化的逻辑。

operations.forEach(op -&gt;
        balances.merge(op.getAccNo(), op.getAmount(), BigDecimal::add)
);
登录后复制

当然结果是正确的,这样简洁的代码心动吗?对于每个操作,add在给定的amount给定accNo

{ 123 = 9.5,456 = - 100 }
登录后复制

ConcurrentHashMap

当我们再延伸到ConcurrentHashMap来,当 Map.merge的出现,和ConcurrentHashMap的结合那是非常的完美的。这样的搭配场景是对于那些自动执行插入或者更新操作的单线程安全的逻辑。

本篇文章到这里就已经全部结束了,更多其他精彩内容可以关注PHP中文网的Java教程视频栏目!

以上就是Map.merge()的详细介绍(附代码)的详细内容,更多请关注php中文网其它相关文章!

智能AI问答
PHP中文网智能助手能迅速回答你的编程问题,提供实时的代码和解决方案,帮助你解决各种难题。不仅如此,它还能提供编程资源和学习指导,帮助你快速提升编程技能。无论你是初学者还是专业人士,AI智能助手都能成为你的可靠助手,助力你在编程领域取得更大的成就。
来源:segmentfault网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
关于CSS思维导图的课件在哪? 课件
凡人来自于2024-04-16 10:10:18
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习
PHP中文网抖音号
发现有趣的

Copyright 2014-2024 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号