Java中我如何去除if...else...语句?

读牛人技术博客 A Java Geek,最开始觉得这样的想法很有创意。提前使用静态代码块把对象存入map容器中,在需要的时候在取。他也有提到可以使用DI的方式把需要的对象提前注入好,但是这两种方式都会造成内存的浪费,因为有一些对象可能是频繁使用,而有些对象用的概率小甚至一次都没有用到,那么这样的方式是不好的。

而且,我们是去除if…else…的语句,这样的方式虽然好像没有了if…else…语句,但是本质上并不是最好的方式,只是提供了一种思维方式。

读《重构 改善既有代码的设计》有一条就是,以多态取代条件表达式。这是才是最本质的解决方式。

这里的去除if…else…语句,不是遇见了if…else…语句就去除。这里是这样描述的:你手上有个条件表达式,它根据对象类型的不同而选择不同的行为。而不是平时编写代码遇见一些普通的条件表达式就去把它去除,不是这样的。

而且,一般来说编程语言都有switch语句去替代if…esle…语句。从性能上看这两个语法也只有非常细微的差别,根本无需关心自己使用了哪个语法。

以多态取代条件表达式
  1. 代码的坏味道一:
double getSpeed() {
   switch_ (_type) {
     case EUROPEAN:
       return getBaseSpeed();
     case AFRICAN:
       return getBaseSpeed() - getLoadFactor() * _numberOfCoconuts;
     case NORWEGIAN_BLUE:
       return (is_Nailed) ? 0 : getBaseSpeed(_voltage);
   }
   throw new RuntimeException("Should be unreachable");
}
  1. 代码的坏味道二:
public Foo getFoo(Bar bar) {

   if (bar instanceof BarA) {
       return new FooA();
   } else if (bar instanceof BarB) {
       return new FooB();
   } else if (bar instanceof BarC) {
       return new FooC();
   } else if (bar instanceof BarD) {
       return new FooD();
   }
   throw new BarNotFoundException();
}

思路

将这个条件表达式的每个分支放进一个子类内的覆写函数中,然后将原始函数声明位抽象函数。

动机(为什么我们要使用多态取代条件表达式?)

多态最根本的好处就是:如果你需要根据对象的不同类型而采取不同的行为,多态使你不必编写明显的条件表达式。

类图

根据代码的坏味道一和二以及类图关系,我们可以得到一个初始版本的代码

Employee

class Employee...
   int payAmount() {
     switch (getType()) {
       case EmployeeType.ENGINEER:
         return _monthlySalary;
       case EmployeeType.SALESMAN:
         return _monthlySalary + _commission;
       case EmployeeType.MANAGER:
         return _monthlySalary + _bonus;
       default:
         throw new RuntimeException("Incorrect Employee");
     }
   } 
   int getType() {
     return _type.getTypeCode();
   }
   private EmployeeType _type;

EmployeeType

abstract class EmployeeType...
   abstract int getTypeCode();

Engineer

class Engineer extends EmployeeType...
  int getTypeCode() {
    return Employee.ENGINEER;
  }

Manager

class Manager extends EmployeeType...
  int getTypeCode() {
    return Employee.MANAGER;
  }

Salesman

class Salesman extends EmployeeType...
  int getTypeCode() {
    return Employee.SALESMAN;
  }

最终结果代码布局

Employee

class Employee...
   int payAmount() {
     return _type.payAmount(this);
   }
   private EmployeeType _type;

EmployeeType

class EmployeeType...
   abstract int payAmount(Employee emp);

Engineer

class Engineer extends EmployeeType...
  int payAmount(Employee emp) {
    return emp.getMonthlySalary();
  }

Manager

calss Manager extends EmployeeType...
  int payAmount(Employee emp) {
    return emp.getMonthlySalary() + emp.getBouns();
  }

Salseman

 class Salseman extends EmployeeType...
     int payAmount(Employee emp) {
       return emp.getMonthlySalary() + emp.getCommission();
     }

理解

a, 把Employee中的原来的属性_monthlySalary,_commission,_bonus变成最终的采用一个函数获取。

b, 注意的地方就是谁组合了谁。这里就是Employee组合了EmployeeType。或者可以理解为谁持有了谁的引用。

c, 其次就是一个继承关系。

小结

由于偶然读到别人技术博客关于if…else…的取代文章,发现这也是一个思路,但是不够那么好,于是记录下《重构 改善既有代码设计》的一个以多态取代条件表达式的范例。

虽然这样确实处理了if…else…语句,但是一旦使用多态取代条件表达式的方式,必定会引入一个继承或者实现体系,其实,这是增加了理解的复杂度。

面向对象的编程,如果增加了扩展性必定会引入设计模式,而一旦引入设计模式必定增加理解的复杂度,因为总有继承体系和实现体系结构,这种结构也是面向对象编程的特色。面向对象编程不一定比面向过程好。在实际的编码过程中,还是需要权衡这两种方式。

本文分享自微信公众号 - 好好学java(SIHAIloveJAVA)

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2019-10-28

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏C语言及其他语言

C++语言的特点 【上】

C++语言是在C语言的基础上发展而来,同时它又支持面向对象的程序设计,它主要具有以下特点:

6710
来自专栏腾讯技术工程官方号的专栏

浅谈 C++ 元编程

? 随着 C++ 11/14/17 标准的不断更新,C++ 语言得到了极大的完善和补充。元编程作为一种新兴的编程方式,受到了越来越多的广泛关注。结合已有文献和...

14530
来自专栏用户5521492的专栏

Spring 复盘 | AOP

Spring AOP 基础 Java 动态代理实现,阅读文章之前,你最好有以下基础:

6830
来自专栏腾讯云TVP

腾讯云TVP李智慧:如何用反应式编程提升系统性能与可用性?

导语 | 没有人能够预言未来,也没有人能够断言未来的编程是什么样,但是我们可以通过过往的编程经验去探寻未来的编程趋势,本文是腾讯云TVP李智慧教你如何用反应式编...

1.9K40
来自专栏大数据成神之路

使用canal-kafka实现数据库增量实时更新

下载安装包: https://github.com/alibaba/canal/releases canal.kafka-1.1.0.tar.gz

21820
来自专栏爱敲代码的猫

java设计模式(1)

1、开闭原则 开闭原则就是说对扩展开放,对修改关闭。在程序需要进行拓展的时候,不能去修改原有的代码,实现一个热插拔的效果。所以一句话概括就是:为了使程序的扩展...

6820
来自专栏爱敲代码的猫

java设计模式(2)-单例设计模式

上一篇推文写了工厂方法模式,包括简单工厂模式、多工厂模式、静态工厂模式、抽象工厂模式,这篇推文记录一下单例设计模式

10840
来自专栏爱敲代码的猫

java设计模式(5)-适配器模式

适配器模式将某个类的接口转换成客户端期望的另一个接口表示,目的是消除由于接口不匹配所造成的类的兼容性问题

4530
来自专栏大内老A

[ASP.NET Core 3框架揭秘] 依赖注入:控制反转

ASP.NET Core框架建立在一些核心的基础框架之上,这些基础框架包括依赖注入、文件系统、配置选项和诊断日志等。这些框架不仅仅是支撑ASP.NET Core...

8330
来自专栏Linux知识积累

创建 REST API 的最佳入门教程

如果你看到这里,你以前可能听说过API 和REST,然后你就会想:“这些都是什么东西?”。也许你已经了解过一些这方面的知识,但却不知道从何入手。在这个教程中,我...

11520

扫码关注云+社区

领取腾讯云代金券

年度创作总结 领取年终奖励