Map中的computeIfAbsent()和computeIfPresent()---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 中才有,大家用的比較少,所以才顯得陌生。
在這裏插入圖片描述

最後附上這個方法改好的代碼


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章