传统的多分支方式(圈复杂度为6):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
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函数式编程的新方式:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
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, 模式>=============> 通过一个数据可以获取一个设计
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:https://www.cnblogs.com/wangxuejian/p/13630024.html