Guava Lists transform 使用及問題

cyril發表於2017-12-14

簡介:Guava 提供的Lists.transform很強大,但在專案中最近的使用遇到了問題,發現還是有坑的,所以本文先是簡單介紹Lists.transform和其使用,後面結合實際的使用過程可能出現的問題來一起看下Lists.transform的實現原理,如有不對的地方歡迎評論討論。

##一:Lists.transform的使用

大家在寫程式碼的過程中肯定會碰到一種狀況,dao中查詢資料庫返回了一個結果集list,其中Result物件中包含了Id,nameStr,msg等欄位,但是上層業務的一些介面引數可能只需要id的結果集list,當然我們可以使用for each迴圈list然後將Id取出依次add到list中,Lists.transform就是幫我們方便的解決這一過程,能夠輕鬆的從一種型別的list轉換為另一種型別的list。  使用方式如下:

  public void listToList(){
    //源list
    List<Result> listResults = Lists.newArrayList(new Result(1,"test1"),new Result(2,"test2"),new Result(3,"test3"));
    //轉換為目標list
    List<String> strLists = Lists.transform(listResults,new Function<Result,String>(){
      @Override
      public String apply(Result result){
        return result.getNameStr();
      }
    });
  }
複製程式碼

##二:Lists.transform使用可能遇到的問題 請看如下程式碼,只是在上面的示例中增加了一行改變listResults中物件屬性的操作和列印:

 @Test
  public void listToList(){
    //源list
    List<Result> listResults = Lists.newArrayList(new Result(1,"test1"),new Result(2,"test2"),new Result(3,"test3"));
    //轉換為目標list
    List<String> strLists = Lists.transform(listResults,new Function<Result,String>(){
      @Override
      public String apply(Result result){
        return result.getNameStr();
      }
    });
    //轉換後目標list列印
    System.out.println("strLists 1 values:");
    for(String str:strLists){
      System.out.println(str+";");
    }
    //修改源list的值
    for(Result result:listResults){
      result.setNameStr("reset");
    }
    //再次列印目標list
    System.out.println("strLists 2 values:");
    for(String str:strLists){
      System.out.println(str+";");
    }
  }
複製程式碼

輸出結果是什麼呢?如果剛開始使用Lists.transform的話,肯定會認為兩次輸出是相同的才對,但實際情況卻不是這樣的,已經轉換後得到的list會受到源list的改動而改變,上面程式碼的輸出結果如下:

strLists 1 values:
test1;
test2;
test3;
strLists 2 values:
reset;
reset;
reset;
複製程式碼

這個坑如果不瞭解就使用Lists.transform的話那麼問題有點大了,會造成很嚴重的問題,我們先來看下Lists.transform是怎樣實現的。

public static <F, T> List<T> transform(List<F> fromList, Function<? super F, ? extends T> function) {
        return (List)(fromList instanceof RandomAccess ? new Lists.TransformingRandomAccessList(fromList, function) : new Lists.TransformingSequentialList(fromList, function));
    }
複製程式碼

可以看到上面的程式碼寫的很清楚,Lists.transform返回的是一個新建立的TransformingSequentialList例項,然後我們再接著往下看

  public ListIterator<T> listIterator(int index) {
            return new TransformedListIterator<F, T>(this.fromList.listIterator(index)) {
                T transform(F from) {
                    return TransformingSequentialList.this.function.apply(from);
                }
            };
        }
複製程式碼

TransformingSequentialList每次遍歷都會從原來的list中遍歷來從新計算得到function ##總結 使用新的開源類庫時最後多做了解,看下原始碼後再使用,這樣就能在合適的場景下合理使用,如果不瞭解的話還是最好先用比較穩妥的方式進行編碼。

相關文章