1. 程式人生 > >Guava學習筆記:Guava新增集合型別-Multiset

Guava學習筆記:Guava新增集合型別-Multiset

  Guava引進了JDK裡沒有的,但是非常有用的一些新的集合型別。所有這些新集合型別都能和JDK裡的集合平滑整合。Guava集合非常精準地實現了JDK定義的介面。Guava中定義的新集合有:
  Multiset
  SortedMultiset
  Multimap
  ListMultimap
  SetMultimap
  BiMap
  ClassToInstanceMap
  Table

  Multiset集合

  Multiset是什麼?顧名思義,Multiset和Set的區別就是可以儲存多個相同的物件。在JDK中,List和Set有一個基本的區別,就是List可以包含多個相同物件,且是有順序的,而Set不能有重複,且不保證順序(有些實現有順序,例如LinkedHashSet和SortedSet等)所以Multiset佔據了List和Set之間的一個灰色地帶:允許重複,但是不保證順序。
  常見使用場景:Multiset有一個有用的功能,就是跟蹤每種物件的數量,所以你可以用來進行數字統計。 常見的普通實現方式如下:

@Test
    public void testWordCount(){
        String strWorld="wer|dffd|ddsa|dfd|dreg|de|dr|ce|ghrt|cf|gt|ser|tg|ghrt|cf|gt|" +
                "ser|tg|gt|kldf|dfg|vcd|fg|gt|ls|lser|dfr|wer|dffd|ddsa|dfd|dreg|de|dr|" +
                "ce|ghrt|cf|gt|ser|tg|gt|kldf|dfg|vcd|fg|gt|ls|lser|dfr";
        String[] words
=strWorld.split("\\|"); Map<String, Integer> countMap = new HashMap<String, Integer>(); for (String word : words) { Integer count = countMap.get(word); if (count == null) { countMap.put(word, 1); } else
{ countMap.put(word, count + 1); } } System.out.println("countMap:"); for(String key:countMap.keySet()){ System.out.println(key+" count:"+countMap.get(key)); } }

  上面的程式碼實現的功能非常簡單,用於記錄字串在陣列中出現的次數。這種場景在實際的開發過程還是容易經常出現的,如果使用實現Multiset介面的具體類就可以很容易實現以上的功能需求:

    public void testMultsetWordCount(){
        String strWorld="wer|dfd|dd|dfd|dda|de|dr";
        String[] words=strWorld.split("\\|");
        List<String> wordList=new ArrayList<String>();
        for (String word : words) {
            wordList.add(word);
        }
        Multiset<String> wordsMultiset = HashMultiset.create();
        wordsMultiset.addAll(wordList);
for(String key:wordsMultiset.elementSet()){ System.out.println(key+" count:"+wordsMultiset.count(key)); } }

  Multiset主要方法

Multiset介面定義的介面主要有:
    add(E element) :向其中新增單個元素
    add(E element,int occurrences) : 向其中新增指定個數的元素
    count(Object element) : 返回給定引數元素的個數
    remove(E element) : 移除一個元素,其count值 會響應減少
    remove(E element,int occurrences): 移除相應個數的元素
    elementSet() : 將不同的元素放入一個Set中
    entrySet(): 類似與Map.entrySet 返回Set<Multiset.Entry>。包含的Entry支援使用getElement()和getCount()
    setCount(E element ,int count): 設定某一個元素的重複次數
    setCount(E element,int oldCount,int newCount): 將符合原有重複個數的元素修改為新的重複次數
    retainAll(Collection c) : 保留出現在給定集合引數的所有的元素
    removeAll(Collectionc) : 去除出現給給定集合引數的所有的元素

  常用方法例項:

@Test
    public void testMultsetWordCount(){
        String strWorld="wer|dfd|dd|dfd|dda|de|dr";
        String[] words=strWorld.split("\\|");
        List<String> wordList=new ArrayList<String>();
        for (String word : words) {
            wordList.add(word);
        }
        Multiset<String> wordsMultiset = HashMultiset.create();
        wordsMultiset.addAll(wordList);
        
        
        //System.out.println("wordsMultiset:"+wordsMultiset);
        
        for(String key:wordsMultiset.elementSet()){
            System.out.println(key+" count:"+wordsMultiset.count(key));
        }
        
        if(!wordsMultiset.contains("peida")){
            wordsMultiset.add("peida", 2);
        }
        System.out.println("============================================");
        for(String key:wordsMultiset.elementSet()){
            System.out.println(key+" count:"+wordsMultiset.count(key));
        }
        
        
        if(wordsMultiset.contains("peida")){
            wordsMultiset.setCount("peida", 23);
        }
        
        System.out.println("============================================");
        for(String key:wordsMultiset.elementSet()){
            System.out.println(key+" count:"+wordsMultiset.count(key));
        }
        
        if(wordsMultiset.contains("peida")){
            wordsMultiset.setCount("peida", 23,45);
        }
        
        System.out.println("============================================");
        for(String key:wordsMultiset.elementSet()){
            System.out.println(key+" count:"+wordsMultiset.count(key));
        }
        
        if(wordsMultiset.contains("peida")){
            wordsMultiset.setCount("peida", 44,67);
        }
        
        System.out.println("============================================");
        for(String key:wordsMultiset.elementSet()){
            System.out.println(key+" count:"+wordsMultiset.count(key));
        }
    }

  輸出:

de count:1
dda count:1
dd count:1
dfd count:2
wer count:1
dr count:1
============================================
de count:1
dda count:1
dd count:1
dfd count:2
peida count:2
wer count:1
dr count:1
============================================
de count:1
dda count:1
dd count:1
dfd count:2
peida count:23
wer count:1
dr count:1
============================================
de count:1
dda count:1
dd count:1
dfd count:2
peida count:45
wer count:1
dr count:1
============================================
de count:1
dda count:1
dd count:1
dfd count:2
peida count:45
wer count:1
dr count:1

  說明:setCount(E element,int oldCount,int newCount): 方法,如果傳入的oldCount和element的不一致的時候,是不能講element的count設定成newCount的。需要注意。

  Multiset不是Map

  需要注意的是Multiset不是一個Map<E,Integer>,儘管Multiset提供一部分類似的功能實現。其它值得關注的差別有:
  Multiset中的元素的重複個數只會是正數,且最大不會超過Integer.MAX_VALUE。設定計數為0的元素將不會出現multiset中,也不會出現elementSet()和entrySet()的返回結果中。
  multiset.size() 方法返回的是所有的元素的總和,相當於是將所有重複的個數相加。如果需要知道每個元素的個數可以使用elementSet().size()得到.(因而呼叫add(E)方法會是multiset.size()增加1).
  multiset.iterator() 會迴圈迭代每一個出現的元素,迭代的次數與multiset.size()相同。 iterates over each occurrence of each element, so the length of the iteration is equal to multiset.size().
  Multiset 支援新增、移除多個元素以及重新設定元素的個數。執行setCount(element,0)相當於移除multiset中所有的相同元素。
  呼叫multiset.count(elem)方法時,如果該元素不在該集中,那麼返回的結果只會是0。

  Multiset的實現 

  Guava提供了Multiset的多種實現,這些實現基本對應了JDK中Map的實現:
  Map                        Corresponding Multiset   Supports null elements
  HashMap                  HashMultiset                      Yes
  TreeMap                   TreeMultiset                       Yes (if the comparator does)
  LinkedHashMap         LinkedHashMultiset             Yes
  ConcurrentHashMap  ConcurrentHashMultiset       No
  ImmutableMap          ImmutableMultiset               No

相關推薦

Guava學習筆記Guava新增集合型別-Multiset

  Guava引進了JDK裡沒有的,但是非常有用的一些新的集合型別。所有這些新集合型別都能和JDK裡的集合平滑整合。Guava集合非常精準地實現了JDK定義的介面。Guava中定義的新集合有:  Multiset  SortedMultiset  Multimap  ListMultimap  SetMult

Guava學習筆記Guava新增集合型別-Multimap

  在日常的開發工作中,我們有的時候需要構造像Map<K, List<V>>或者Map<K, Set<V>>這樣比較複雜的集合型別的資料結構,以便做相應的業務邏輯處理。例如: import java.util.ArrayList; import java

Guava學習筆記Guava新增集合型別-Bimap

  BiMap提供了一種新的集合型別,它提供了key和value的雙向關聯的資料結構。  通常情況下,我們在使用Java的Map時,往往是通過key來查詢value的,但是如果出現下面一種場景的情況,我們就需要額外編寫一些程式碼了。首先來看下面一種表示標識序號和檔名的map結構。 @Test

Guava學習筆記Guava集合-Table等

  Table  當我們需要多個索引的資料結構的時候,通常情況下,我們只能用這種醜陋的Map<FirstName, Map<LastName, Person>>來實現。為此Guava提供了一個新的集合型別-Table集合型別,來支援這種資料結構的使用場景。Table支援“row”和“

Guava學習筆記Guava cache

  快取,在我們日常開發中是必不可少的一種解決效能問題的方法。簡單的說,cache 就是為了提升系統性能而開闢的一塊記憶體空間。   快取的主要作用是暫時在記憶體中儲存業務系統的資料處理結果,並且等待下次訪問使用。在日常開發的很多場合,由於受限於硬碟IO的效能或者我們自

Guava學習筆記Immutable(不可變)集合

  不可變集合,顧名思義就是說集合是不可被修改的。集合的資料項是在建立的時候提供,並且在整個生命週期中都不可改變。   為什麼要用immutable物件?immutable物件有以下的優點:     1.對不可靠的客戶程式碼庫來說,它使用安全,可以在未受信任的類庫中安全的使用這些物件    2.執行緒安全

Guava學習筆記Optional優雅的使用null

asset 不包含 你在 rgs 命名 靜態 不清晰 ces throw 在我們學習和使用Guava的Optional之前,我們需要來了解一下Java中null。因為,只有我們深入的了解了null的相關知識,我們才能更加深入體會領悟到Guava的Optional設計和使用上

Guava學習筆記Preconditions優雅的檢驗引數

在日常開發中,我們經常會對方法的輸入引數做一些資料格式上的驗證,以便保證方法能夠按照正常流程執行下去。對於可預知的一些資料上的錯誤,我們一定要做事前檢測和判斷,來避免程式流程出錯,而不是完全通過錯誤處理來保證流程正確執行,畢竟錯誤處理是比較消耗資源的方式。在平常情況下我們對引數的判斷都需要自己來逐個寫方法

Guava學習筆記EventBus

  EventBus是Guava的事件處理機制,是設計模式中的觀察者模式(生產/消費者程式設計模型)的優雅實現。對於事件監聽和釋出訂閱模式,EventBus是一個非常優雅和簡單解決方案,我們不用建立複雜的類和介面層次結構。   Observer模式是比較常用的設計模式之一,雖然有時候在具體程式碼裡,它不一定

Guava學習筆記複寫的Object常用方法

  在Java中Object類是所有類的父類,其中有幾個需要override的方法比如equals,hashCode和toString等方法。每次寫這幾個方法都要做很多重複性的判斷, 很多類庫提供了覆寫這幾個方法的工具類, Guava也提供了類似的方式。下面我們來看看Guava中這幾個方法簡單使用。   e

Guava學習筆記簡化異常處理的Throwables類

  有時候, 當我們我們捕獲異常, 並且像把這個異常傳遞到下一個try/catch塊中。Guava提供了一個異常處理工具類, 可以簡單地捕獲和重新丟擲多個異常。例如: import java.io.IOException; import org.junit.Test; import com.goog

Guava學習筆記Range

  在Guava中新增了一個新的型別Range,從名字就可以瞭解到,這個是和區間有關的資料結構。從Google官方文件可以得到定義:Range定義了連續跨度的範圍邊界,這個連續跨度是一個可以比較的型別(Comparable type)。比如1到100之間的整型資料。   在數學裡面的範圍是有邊界和無邊界之分

Guava學習筆記Ordering犀利的比較器

  Ordering是Guava類庫提供的一個犀利強大的比較器工具,Guava的Ordering和JDK Comparator相比功能更強。它非常容易擴充套件,可以輕鬆構造複雜的comparator,然後用在容器的比較、排序等操作中。   本質上來說,Ordering 例項無非就是一個特殊的Comparat

Go語言學習筆記(三) 集合型別

陣列 陣列中包含的每個資料被稱為陣列元素(element),一個數組包含的元素個數被稱為陣列的長度。 [32]byte // 長度為32的陣列,每個元素為一個位元組 [2*N] struct { x, y int32 } // 複雜型別陣列 [1000]*float64 // 指標陣列 [3

ElasticSearch 6.x 學習筆記12.欄位型別

12.1 欄位型別概述 一級分類 二級分類 具體型別 核心型別 字串型別 string,text,keyword 整數型別 integer,long,

go學習筆記2.變數、型別、常量

基本型別 1.基本型別列表 複製程式碼程式碼如下: 型別        長度     說明 bool         1      true/false,預設false, 不能把非0值當做true(不用數字代表true/false) byte        

Guava學習筆記(二)基礎(Joiner,Objects,Splitter及Strings)

nonnull obj expect null dto 字符 情況 core cte 添加Maven依賴 JoinerTest import com.google.common.base.Joiner; import org.junit.Assert; import org

Guava學習筆記(二)Google Guava (瓜娃)的API快速熟悉使用

1,大綱 讓我們來熟悉瓜娃,並體驗下它的一些API,分成如下幾個部分: IntroductionGuava Collection APIGuava Basic UtilitiesIO APICache API 2,為神馬選擇瓜娃? 瓜娃是java API蛋糕上的冰

Guava新增集合型別的例子

Multimap ArrayListMultimap實現 Multimap也稱為Multihash,它是Map的變體,它允許用相同的鍵關聯多個值或者物件。顧名思義,Multimap基於儲存鍵值對的Map介面。它允許重複的鍵值對存在。例如,Amaury V

Guava學習筆記目錄

筆記 htm 目錄 join cnblogs string com log href 源碼 Guava學習筆記(一):Maven Guava學習筆記(二):基礎(Joiner,Objects,Splitter及Strings) Guava學習筆記(三):集合 Guava學