Java8特性使用Function代替分支語句
傳統的多分支方式(圈複雜度為6):
public String order(String type) { if ("1".equals(type)) { return "1"; } else if ("2".equals(type)) { return "2"; } else if ("3".equals(type)) { return "3"; } else if ("4".equals(type)) { return "4"; } else if ("5".equals(type)) { return "5"; } else { return "none"; } }
使用Function函數語言程式設計的新方式:
private Map<String,Function<String,String>> map = new HashMap<>(); public String order2(String type){ init(); Function<String,String> fun = map.get(type); String result = fun.apply(type); return result; } public void init() { map.put("1",a -> { System.out.println("執行1分支"); return a + "1"; }); map.put("2",a -> { System.out.println("執行2分支"); return a + "22"; }); map.put("3",a -> { System.out.println("執行3分支"); return a + "333"; }); map.put("4",a -> { System.out.println("執行4分支"); return a + "4444"; }); map.put("5",a -> { System.out.println("執行5分支"); return a + "55555"; }); }
圈複雜度為0;
總結:
1.Function是一個介面,它完整的樣子是這樣: Function<T,R>,其中T表示接受一個T型別的入參,R表示返回一個R型別的出參,注意: 既然入參和出參都是一個泛型,那就說明可以適用於任何種類的方法,如果方法需要多個入參,可以把它們封裝成T型別的入參VO,如果出參也需要有多個引數,也可以把它們封裝為一個R型別的出參VO.
2. apply方法是function函式的統一執行標誌性方法,它接受第一條提到的T t,返回第一條提到的R r,不管function函式體有多複雜,統統按apply執行,就像所有的執行緒統統用start開始執行.
3. 以前在有很多if-else的場合時,也想過把決定分支的入參(用type表示)和所執行的程式碼建立對映關係,但那時候因為不懂Function,心裡想Map裡面只能裝資料型別,像分支邏輯這種不是資料型別,沒法裝.如今學習了Function,發現它把程式碼邏輯也封裝得了,正好滿足了我的設想.真的太好用了.
4. 以前用if-else執行多分支時,走入不同分支還需要一個個對照,現在用了函數語言程式設計,直接根據map查對映關係就找到了,很是方便.
最後總結:
以後遇到if-else較多的場合,可以考慮使用Function函式代替
普通的方法是將一種資料型別作為引數,而Function方法是將一種方法或表示式作為引數.
=======================補充: 與策略模式的對比=======================
策略模式也是為了解決分支過多問題,此外它還解決另一個大問題: 避免程式碼侵入,新增策略不需要修改原有策略程式碼.此處主要對比第一點: 解決分支過多
策略模式通過兩層結構,完美實現了傳入不同的策略,執行不同的方案,這與if-else的目標是一致的,當然與Function的目標也是一致的.
傳入1,選擇策略1,執行策略1
傳入2,選擇策略2,執行策略2
如何實現傳入N,就能選擇策略N呢? 使用對映表Map<String,Strategy>,這點和Function的思想是一樣,只不過策略模式通過型別獲取封裝好的策略例項,而Function通過型別獲取封裝好的策略方法.
從廣義上說,if-else,Function,策略模式,它們三者都是策略,解決的都是分支問題,只不過粒度由小到大,重量級不同而已.
如何選擇呢?
能用if-else就不用Function,能用Function就不用策略模式
簡單的,就用輕量級的
複雜的,就用重量級的.
===============題外思考: Map的三種用法================
- 初級: Map<String,普通資料型別>======>通過一個數據可以獲取另一個數據
- 中級: Map<String,函式/表示式>=======> 通過一個數據可以獲取一個方法
- 高階: Map<String,模式>=============> 通過一個數據可以獲取一個設計
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。