Java进阶篇— JDK8新增的Map方法

Posted 普通网友

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java进阶篇— JDK8新增的Map方法相关的知识,希望对你有一定的参考价值。

JDK8最大的新特性便是增加反应式流编程。作为最常用的数据结构Map。它有什么变化呢?我们能不能更加方便的使用呢?

在JDK8中Map的一些方法可以使用Lambda表达式了。而Lambda表达式是就是匿名内部类的简写。实际上可以理解为Map的参数可以是一段代码了。即行为参数化

1. 什么叫做行为参数化

以Map的遍历为例:

匿名内部类的调用方式:

    @Test
    public void testMap() 
        Map<String, String> map = new HashMap<>();
        map.forEach(new BiConsumer<String, String>() 
            @Override
            public void accept(String key, String value) 
                System.out.println(key+":"+value);
            
        );
    

Lambda表达式的调用方式:

    @Test
    public void testMap() 
        Map<String, String> map = new HashMap<>();
        map.forEach((key,value)-> System.out.println(key+":"+value));
    

源码:java.util.Map#forEach

    default void forEach(BiConsumer<? super K, ? super V> action) 
        Objects.requireNonNull(action);
        //遍历Map,获取key和value
        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);
        
    

foreach()方法的参数是一个对象。我们传入不同的对象,该方法有不同的处理逻辑。
实际上我们传入的是一个策略对象

Map的新特性,其实便是策略模式的一种运用,需要注意的是,当使用Lambda表达式作为参数简写匿名内部类,实际上传入的依旧是对象。匿名内部类逻辑,将在调用方法某处执行。

[详解——策略设计模式]

策略模式属于对象的行为模式,其用意是定义一族算法,把他们封装起来(称为策略),在运算的时候选择一种策略。
策略模式的重心不在于如何实现算法,而是如何组织,调用这些算法,从而让程序结构更加灵活,具有更好的维护性和扩展性。

2. Map常用方法

Map在项目中一般作为缓存使用,使用比较多的方法有遍历方法、put方法或get方法。

我们一般的操作缓存:若Map中存在,我们直接返回用户,若map不存在,则先将其加入到缓存中,在返回。若下列代码所示:

    public Cache getCache(String name) 
        Cache cache = this.cacheMap.get(name);
        if (cache != null) 
            return cache;
        
        else 
            // Fully synchronize now for missing cache creation...
            synchronized (this.cacheMap) 
                cache = this.cacheMap.get(name);
                if (cache == null) 
                    cache = getMissingCache(name);
                    if (cache != null) 
                        cache = decorateCache(cache);
                        this.cacheMap.put(name, cache);
                        updateCacheNames(name);
                    
                
                return cache;
            
        
    

在JDK8的Map中,有没有好的方式去实现上述功能?

  1. 遍历方法
    @Test
    public void testMap() 
        Map<String, String> map = new HashMap<>();
        map.forEach((key,value)-> System.out.println(key+":"+value));
    

  1. 【获取并存储】putIfAbsent (如果不存在时进行存储,但返回null)

若映射存在,返回value;若映射不存在,存储value返回null。

使用方式:

    @Test
    public void testMap() 
        Map<String, String> map = new HashMap<>();
        String v1 = map.putIfAbsent("aa", "v1");
        String v2 = map.putIfAbsent("aa", "v2");
        System.out.println("key不存在时,返回的值:" + v1);
        System.out.println("key存在时,返回的值:" + v2);
        System.out.println("此时,key的映射:" + map.get("aa"));
    

返回结果

key不存在时,返回的值:null
key存在时,返回的值:v1
此时,key的映射:v1

  1. 【获取并存储】computeIfAbsent:(如果不存在时进行存储,返回存储的值)

可以看到,匿名内部类的参数实际上是Key的值。

    @Test
    public void testMap() 
        Map<String, String> map = new ConcurrentHashMap<>();
        String v1 = map.computeIfAbsent("aa", K->K+"v1");
        String v2 = map.computeIfAbsent("aa", K->K+"v2");
        System.out.println("key不存在时,返回的值:" + v1);
        System.out.println("key存在时,返回的值:" + v2);
        System.out.println("此时,key的映射:" + map.get("aa"));
    

返回结果:

key不存在时,返回的值:aav1
key存在时,返回的值:aav1
此时,key的映射:aav1

  1. 获取value(若key存在,对key和value操作得到newValue,替换原值)
    @Test
    public void testMap() 
        Map<String, String> map = new HashMap<>();
        map.computeIfPresent("aa", (k,v)->getValue(k,v));
    

  1. 获取value(若key不存在,返回默认值)
    @Test
    public void testMap() 
        Map<String, String> map = new HashMap<>();
        map.getOrDefault("aa",new String("bb"));
    

以上是关于Java进阶篇— JDK8新增的Map方法的主要内容,如果未能解决你的问题,请参考以下文章

Java集合中的Map接口

JDK8

JDK7新增Objects方法介绍(同时包含JDK8新添加的方法)

##(⊙o⊙)值得收藏的JavaSE万字进阶版(⊙o⊙)##JavaSE 高级反射-动态代理-设计模式-JVM篇

Java 集合学习笔记:IterableIterator

JDK8新特性之接口中新增的方法(默认方法和静态方法的作用)