<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          Java 8 中 Map 騷操作之 merge() 的用法

          共 11525字,需瀏覽 24分鐘

           ·

          2021-10-02 07:42

          程序員的成長之路
          互聯(lián)網(wǎng)/程序員/技術(shù)/資料共享 
          關(guān)注


          閱讀本文大概需要 3.5 分鐘。


          Java 8 最大的特性無異于更多地面向函數(shù),比如引入了 lambda等,可以更好地進(jìn)行函數(shù)式編程。前段時間無意間發(fā)現(xiàn)了 map.merge() 方法,感覺還是很好用的,此文簡單做一些相關(guān)介紹。首先我們先看一個例子。

          merge() 怎么用?


          假設(shè)我們有這么一段業(yè)務(wù)邏輯,我有一個學(xué)生成績對象的列表,對象包含學(xué)生姓名、科目、科目分?jǐn)?shù)三個屬性,要求求得每個學(xué)生的總成績。加入列表如下:



              private List<StudentScore> buildATestList() {
                  List<StudentScore> studentScoreList = new ArrayList<>();
                  StudentScore studentScore1 = new StudentScore() {{
                      setStuName("張三");
                      setSubject("語文");
                      setScore(70);
                  }};
                  StudentScore studentScore2 = new StudentScore() {{
                      setStuName("張三");
                      setSubject("數(shù)學(xué)");
                      setScore(80);
                  }};
                  StudentScore studentScore3 = new StudentScore() {{
                      setStuName("張三");
                      setSubject("英語");
                      setScore(65);
                  }};
                  StudentScore studentScore4 = new StudentScore() {{
                      setStuName("李四");
                      setSubject("語文");
                      setScore(68);
                  }};
                  StudentScore studentScore5 = new StudentScore() {{
                      setStuName("李四");
                      setSubject("數(shù)學(xué)");
                      setScore(70);
                  }};
                  StudentScore studentScore6 = new StudentScore() {{
                      setStuName("李四");
                      setSubject("英語");
                      setScore(90);
                  }};
                  StudentScore studentScore7 = new StudentScore() {{
                      setStuName("王五");
                      setSubject("語文");
                      setScore(80);
                  }};
                  StudentScore studentScore8 = new StudentScore() {{
                      setStuName("王五");
                      setSubject("數(shù)學(xué)");
                      setScore(85);
                  }};
                  StudentScore studentScore9 = new StudentScore() {{
                      setStuName("王五");
                      setSubject("英語");
                      setScore(70);
                  }};

                  studentScoreList.add(studentScore1);
                  studentScoreList.add(studentScore2);
                  studentScoreList.add(studentScore3);
                  studentScoreList.add(studentScore4);
                  studentScoreList.add(studentScore5);
                  studentScoreList.add(studentScore6);
                  studentScoreList.add(studentScore7);
                  studentScoreList.add(studentScore8);
                  studentScoreList.add(studentScore9);

                  return studentScoreList;
              }
          我們先看一下常規(guī)做法:
                  ObjectMapper objectMapper = new ObjectMapper();
                  List<StudentScore> studentScoreList = buildATestList();

                  Map<String, Integer> studentScoreMap = new HashMap<>();
                  studentScoreList.forEach(studentScore -> {
                      if (studentScoreMap.containsKey(studentScore.getStuName())) {
                          studentScoreMap.put(studentScore.getStuName(), 
                                              studentScoreMap.get(studentScore.getStuName()) + studentScore.getScore());
                      } else {
                          studentScoreMap.put(studentScore.getStuName(), studentScore.getScore());
                      }
                  });

                  System.out.println(objectMapper.writeValueAsString(studentScoreMap));

          // 結(jié)果如下:
          // {"李四":228,"張三":215,"王五":235}
          然后再看一下 merge() 是怎么做的:
                  Map<String, Integer> studentScoreMap2 = new HashMap<>();
                  studentScoreList.forEach(studentScore -> studentScoreMap2.merge(
                    studentScore.getStuName(),
                    studentScore.getScore(),
                    Integer::sum));

                  System.out.println(objectMapper.writeValueAsString(studentScoreMap2));

          // 結(jié)果如下:
          // {"李四":228,"張三":215,"王五":235}


          merge() 簡介


          merge() 可以這么理解:它將新的值賦值到 key (如果不存在)或更新給定的key 值對應(yīng)的 value,其源碼如下:


              default V merge(K key, V value, BiFunction<? super V, ? super V, ? extends V> remappingFunction{
                  Objects.requireNonNull(remappingFunction);
                  Objects.requireNonNull(value);
                  V oldValue = this.get(key);
                  V newValue = oldValue == null ? value : remappingFunction.apply(oldValue, value);
                  if (newValue == null) {
                      this.remove(key);
                  } else {
                      this.put(key, newValue);
                  }

                  return newValue;
              }

          我們可以看到原理也是很簡單的,該方法接收三個參數(shù),一個 key 值,一個 value,一個 remappingFunction ,如果給定的key不存在,它就變成了 put(key, value) 。


          但是,如果 key 已經(jīng)存在一些值,我們 remappingFunction 可以選擇合并的方式,然后將合并得到的 newValue 賦值給原先的 key。


          使用場景


          這個使用場景相對來說還是比較多的,比如分組求和這類的操作,雖然 stream 中有相關(guān) groupingBy() 方法,但如果你想在循環(huán)中做一些其他操作的時候,merge() 還是一個挺不錯的選擇的。


          其他


          除了 merge() 方法之外,我還看到了一些Java 8 中 map 相關(guān)的其他方法,比如 putIfAbsent 、compute() 、computeIfAbsent() 、computeIfPresent,這些方法我們看名字應(yīng)該就知道是什么意思了,故此處就不做過多介紹了,感興趣的可以簡單閱讀一下源碼(都還是挺易懂的),這里我們貼一下 compute()(Map.class) 的源碼,其返回值是計算后得到的新值:


              default V compute(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction) {
                  Objects.requireNonNull(remappingFunction);
                  V oldValue = this.get(key);
                  V newValue = remappingFunction.apply(key, oldValue);
                  if (newValue == null) {
                      if (oldValue == null && !this.containsKey(key)) {
                          return null;
                      } else {
                          this.remove(key);
                          return null;
                      }
                  } else {
                      this.put(key, newValue);
                      return newValue;
                  }
              }


          總結(jié)


          本文簡單介紹了一下 Map.merge() 的方法,除此之外,Java 8 中的 HashMap 實(shí)現(xiàn)方法使用了 TreeNode 和 紅黑樹,在源碼閱讀上可能有一點(diǎn)難度,不過原理上還是相似的,compute() 同理。所以,源碼肯定是要看的,不懂的地方多讀多練自然就理解了。


          作者:LQ木頭
          juejin.im/post/5d9b455ae51d45782b0c1bfb


          參考:
          https://www.jianshu.com/p/68e6b30410b0
          測試代碼地址:
          https://github.com/lq920320/algorithm-java-test/blob/master/src/test/java/other/MapMethodsTest.java
          <END>

          推薦閱讀:

          國產(chǎn)Linux發(fā)行版再添一員,操作界面不輸蘋果!

          老大:用腳本實(shí)現(xiàn)Nginx日志每天切分的功能?我:脫了褲子放屁!

          最近面試BAT,整理一份面試資料《Java面試BATJ通關(guān)手冊》,覆蓋了Java核心技術(shù)、JVM、Java并發(fā)、SSM、微服務(wù)、數(shù)據(jù)庫、數(shù)據(jù)結(jié)構(gòu)等等。

          獲取方式:點(diǎn)個「在看」,點(diǎn)擊上方小卡片,進(jìn)入公眾號后回復(fù)「面試題」領(lǐng)取,更多內(nèi)容陸續(xù)奉上。

          朕已閱 

          瀏覽 19
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  青青激情视频免费观看在线观看 | 日韩一区二区三区四区久久久精品有吗 | 婷婷黄色伊人 | 一区二区三区视频在线观看 | 亚洲无码在线精品 |