先上代碼:
//g=cache(f)public <T,R> Function<T,R> cache(Function<T,R> f){final Map<T,R> cache=new HashMap<>();Function<T,R> g=t->{if(cache.containsKey(t)){System.out.println("cached t:"+t);return cache.get(t);}System.out.println("not cached t:"+t);R r=f.apply(t);cache.put(t, r);return r;};return g;
// return new Function<T,R>(){
// final Map<T,R> cache=new HashMap<>();
//
// @Override
// public R apply(T t) {
// if(cache.containsKey(t)){
// System.out.println("cached t:"+t);
// return cache.get(t);
// }
// R r=f.apply(t);
// System.out.println("not cached t:"+t);
// cache.put(t, r);
// return r;
// }
//
// };}
這是網上的一篇文章后面留的習作,題目是傳入一個函數f
,返回一個等效的cache版本函數g
。題目本身沒什么,只是在我實現的過程中對兩種方法有一些感想,列在下面以作備忘。
- 要清楚自由變量和限定變量的區別
- 僅從效果來看lambda寫起來簡潔,但不如內部類靈活
- 匿名內部類編譯后會生成一個頂層類,編譯器自動生成的構造函數里,第一個參數必定是包裝類的引用,其余的參數依次是捕獲的自由變量。
- lambda不會生成另外的類(沒有發現class文件)
- lambda對應invokedaynamic指令
一些擴展閱讀的博客文章
通過字節碼分析JDK8中Lambda表達式編譯及執行機制
java8 lambda學習筆記之編譯與運行過程
Java8學習筆記(4) -- Lambda表達式實現方式
Lambda表達式