Java 8的这个新特性,你用了吗?

时间:2022-07-22
本文章向大家介绍Java 8的这个新特性,你用了吗?,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
背景

今天写的一个方法,如下,实现的功能就是将数据库中的表字段获取,存在map 中,将前端传过来的数据存在map 中。

/**
     * 给表中添加数据
     * @param object   数据
     */
    protected void fillData(String type,Object object){
        JSONArray jsonArr = changeObjetToJsonArray(object);
        if(jsonArr==null || jsonArr.isEmpty()){
            return;
        }
        //遍历jsonarray
        GetDateByTableName getDateByTableName=new GetDateByTableName(testNumber,imsi);
        for(int i=0;i<jsonArr.size();i++){
            List<String[]> rowList=new ArrayList<>();
            JSONObject jsonItem=jsonArr.optJSONObject(i);
            if(jsonItem!=null){
                String typeName=BaseDataPreHandle.typeNamePreHandle(type, jsonItem);
                TypeForTableName typeForTableName=Enum.valueOf(TypeForTableName.class,typeName.toUpperCase());
                String tableName=typeForTableName.getTableName();
                /**mapTableColumns.computeIfAbsent(tableName,k -> getFieldByTableName(tableName));*/
                if(mapTableColumns.get(tableName)==null){
                    mapTableColumns.put(tableName,getFieldByTableName(tableName));
                }
                String[] row= getDateByTableName.getDatasbyName(tableName,jsonItem,mapTableColumns.get(tableName));
                if(mapTableValues.get(tableName)!=null){
                    rowList=mapTableValues.get(tableName);
                }
                rowList.add(row);
                mapTableValues.put(tableName,rowList);
            }
        }

    }

异味?

然后我写完用sonarLint 检测发现两个异味,竟然提示我不要这么写,而是提示使用Map.computeIfAbsent()方法,当时很纳闷

使用了lambda表达式,然后我就看了一下computeIfAbsent()源码,如下

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;
    }

我看这个方法的意思是如果get(key)为null,那么就创建一个value值,这个value值就是lambda表达式的返回值,并且这个返回值不为null,才会将这个这个值存到map对应的key 中。再来看一下我最开始写的代码的需求

if(mapTableColumns.get(tableName)==null){
 mapTableColumns.put(tableName,getFieldByTableName(tableName));
 }

如果map中这个key 对应的值不存在,就将新值赋给这个key。和上面computeIfAbsent() 的方法是一致的,因为我可以保证新值不为null。简单的说就是,map中key值存在就不更新,不存在就更新。所以修改为如下:

mapTableColumns.computeIfAbsent(tableName,k -> getFieldByTableName(tableName));

再接着看我后面的代码,单独拿出来

String[] row= getDateByTableName.getDatasbyName(tableName,jsonItem,mapTableColumns.get(tableName));
     if(mapTableValues.get(tableName)!=null){
          rowList=mapTableValues.get(tableName);
      }
      rowList.add(row);
      mapTableValues.put(tableName,rowList);

这个的意思是,有一个list集合,如果map.get(key)不为空,就将这个值赋给list集合,如果这个值为空就跳过,之后,将list添加一条数据,然后将新的list集合添加到这个map对应的key 值中。简单的说就是更新map 对应key 的值。不管map.get(key)是否为空。

那这样可以使用computeIfAbsent()方法吗,显然是不可以的,因为源代码很清楚,只有当get(key)为空时才会进行更新,不为空的话直接就返回了。

所以computeIfAbsent() 方法的用法总结:

 只有在当前 Map 中 key 对应的值不存在或为 null 时
 才调用 mappingFunction
 并在 mappingFunction 执行结果非 null 时
 将结果跟 key 关联.
 mappingFunction 为空时 将抛出空指针异常

修改

那么问题来了,现在我的这段代码snoarLint 检测出异味,改怎么修改呢? 这让我想到了computeIfPresent()方法,看了一下源码

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;
        }
    }

看源码的意思是:当get(key)值存在时,并且新值不为空,就会更新。否则就返回null。

这与我的需求不相符啊还是,我的需求是,不管key值是否为空,都需要进行更新,我这个更新还是在原来的值上增加数据。

那现在怎样消除异味呢?

实在没办法我使用了computeIfAbsent()和computeIfPresent()两种方法结合。先用computeIfAbsent()方法,保证一定可以获取到get(key) 再用computeIfPresent() 方法,保证一定对get(key)更新。如下:

 String[] row= getDateByTableName.getDatasbyName(tableName,jsonItem,mapTableColumns.get(tableName));
 
 mapTableValues.computeIfAbsent(tableName,k->new ArrayList<String[]>());
 
 List<String[]> rowList=mapTableValues.get(tableName);
 rowList.add(row);
 
 mapTableValues.computeIfPresent(tableName,(k,v) -> rowList);

最后再检测,消除异味。虽然这段代码看上去没有原来那么好理解,但是我觉得还是个人对这个方法熟悉与否,put和get方法大家比较熟悉,所以一眼就能看懂,但是computeIfAbsent和computeIfPresent 方法在java 8 中才有,大家用的比较少,所以才显得陌生。

最后附上这个方法改好的代码

/**
     * 给表中添加数据
     * @param object   数据
     */
    protected void fillData(String type,Object object){
        JSONArray jsonArr = changeObjetToJsonArray(object);
        if(jsonArr==null || jsonArr.isEmpty()){
            return;
        }
        //遍历jsonarray
        GetDateByTableName getDateByTableName=new GetDateByTableName(testNumber,imsi);
        for(int i=0;i<jsonArr.size();i++){
            JSONObject jsonItem=jsonArr.optJSONObject(i);
            if(jsonItem!=null){
                String typeName=typeNamePreHandle(type, jsonItem);
                TypeForTableName typeForTableName=Enum.valueOf(TypeForTableName.class,typeName.toUpperCase());
                String tableName=typeForTableName.getTableName();

                mapTableColumns.computeIfAbsent(tableName,k -> getFieldByTableName(tableName));
               
                String[] row= getDateByTableName.getDatasbyName(tableName,jsonItem,mapTableColumns.get(tableName));
                mapTableValues.computeIfAbsent(tableName,k->new ArrayList<String[]>());
                List<String[]> rowList=mapTableValues.get(tableName);
                rowList.add(row);
                mapTableValues.computeIfPresent(tableName,(k,v) -> rowList);
            }
        }

    }