过多if-else分支的优化

 

超过3个就应该去优化,说if-else过多的分支可以使用switch或者责任链模式等等方式来优化。确实,这是一个小问题,不过我们还是可以整理一下这个小问题的重构方式。

为什么要优化?

你没有看错。这是要放在第一条谈论的。

有许多人会说,叠起来一堆if-else分支,代码就不优雅了。可是,怎样去定义“优雅”的概念呢?再退一步说,即便不“优雅”,又有什么问题?

对于这样一段再普通不过的代码:

1
2
3
4
5
6
7
8
int code;
if("Name".equals(str))
    code = 0;
else if("Age".equals(str))
    code = 1;
else if("Address".equals(str))
    code = 2;
...

可以有好多种重构方式,但是使用这样的代码,虽然简陋,但在大多数情况下,并不会影响什么,比如,对可维护性没有影响。当然,如果你发现其中确有不好的一面,那就要考虑重构它。换言之,通常你首先要说出某段代码的问题(比如,你觉得这段代码不符合开闭原则,因为你希望保持这段代码闭合稳定),那么才去存在重构的必要,而不要总是使用“优雅”和“简洁”搪塞疑问。几乎所有的书上都说要写出优雅的、简洁的代码,这本身无可厚非,但是事物需要使用自己的判断,可不要被习惯性地洗了脑。

在我前一家公司,是典型的通讯和传统软件的公司,代码质量普遍不错,但是很多时候,会看到许许多多不够优雅的代码——也许你觉得不够简洁、美观,但是下代码严谨、清晰,我觉得这就很好。反之,某一些精巧的设计,可能会带来可阅读性和可理解性下降的问题。

寻找代替分支判断的方式

接下去我们再来考虑怎么样去重构优化过多的if-else分支。

程序逻辑最基本的组成就是分支、判断和循环。而过多if-else正是由于在某一个变化的点上,有许多判断条件和结果分支造成的。所以最基本的解决办法就是把多个判断条件合成一个,也就是把若干个分支合成一个。

但是在大多数情况下,条件判断的分支都是无法合并的。所以,我们需要把这个变化点通过别的途径封装起来,而不是采用if-else。

1. 用一个Map可以做到,if-else的变化点使用Map的get方法来代替:

1
2
3
4
5
6
Map typeCodeMap = new HashMap();
typeCodeMap.put("Name", 0);
typeCodeMap.put("Age", 1);
typeCodeMap.put("Address", 2);
...
int code = typeCode.get(type);

2. 枚举:

1
2
3
4
5
6
7
8
9
10
11
public enum Codes {
    Name(0), Age(1), Address(2);
     
    public int code;
    Codes(int code){
        this.code = code;
    }
}
 
//使用:
int code = Codes.valueOf(str).code;

3. 多态:

1
2
ICode iCode = (ICode)Class.forName("com.xxx." + str).newInstance();
int code = iCode.getCode();

当然,如果仅考虑从String转向int这样的转换,用这样的方式来简化分支判断逻辑,这个方式、这个例子不是很恰当。当然,这样的方式经常被用来做从字符串到具体对象的转换。

还有一些朋友说的这个模式那个模式来解决多if-else的问题,这些都是正确的,当然本质上也无一例外基于多态来实现的,所以我就不提及了。这些都不错,至少比那些老说用switch来代替if-else的有价值多了 :)

最后,对于如此小的一个问题,我要补充说明的一点是,看不得大片if-else和看不得大片new关键字一样,我觉得这是许多Java程序员的既有观念或者说习惯,甚至通病——这并不好。Java最有价值的地方不是它的语义语法也不是它的虚拟机跨平台和有多高性能,而在于它的社区它的无比丰富的类库,在于使用它的人可以从设计上和宏观上去思考问题。但是Java程序员,也包括我在内,很容易把这条路走得过于极端,比如遍地的Factory,比如漫山遍野的配置,比如永远也不会被复用的可复用代码,比如永远也不会被扩展的可扩展代码,还比如从前到后由内到外的分层,一层又一层。相对于这些方面无止境的追求,我们还是专注于要解决的问题,多写一些清晰可用的代码吧。

最新文章

  1. PHP判断sql语句是否执行成功
  2. VS2015 调试Web项目 遭遇 HTTP 错误 500.23 - Internal Server Error
  3. Json数据中的特殊字符处理
  4. Round函数
  5. NSString几个函数
  6. 【原创】C#搭建足球赛事资料库与预测平台(5) 赔率数据表设计1
  7. 基础学习day06---面向对象二---static,类的初始化和调用顺序、单例模式
  8. 中科院Oracle 10G 数据库系统培训视频教程(828MB )
  9. Java NIO通信框架在电信领域的实践
  10. 【python】【转】Python生成随机数的方法
  11. eclipse format的时候如何让@param后不换行
  12. Oracle自动备份脚本
  13. 物联网 开发板 基于ESP8266
  14. 升级版本后报这个异常 : org.springframework.beans.factory.NoUniqueBeanDefinitionException
  15. C++构造函数(一)
  16. chrome开发工具指南(一)
  17. 关于a标签颜色的探索
  18. laravel 数据库获取值的常用方法
  19. GitLab Shell如何通过SSH工作
  20. Day Four

热门文章

  1. jQuery val()方法及valHooks源码解读
  2. 关于.NET .cs后台提示并进行页面跳转代码
  3. Flask 学习系列(一) -登录
  4. jQuery的基本使用及选择器和筛选器
  5. jQuery deferred应用之ajax详细源码分析(二)
  6. Sublime Text 3安装AngularJS插件
  7. git 设置了ssh key 还是需要输入账户和密码
  8. Xilinx FPGA结构
  9. Windows 8.1 explorer.exe 出错 “Application Hang”
  10. iPhone Tutorials