使用面向对象和功能性方法来重构Java应用

译文 精选
开发 前端
本文将和您讨论如何使用面向对象和功能性两种方法,来实现更好的设计与 Java 应用重构,并以继承和功能性接口的示例形式,展示了两者在实现上的区别。

译者 | 陈峻

审校 | 孙淑娟

在重构方面,Java主要有两种主要方法,即:面向对象(object-oriented)和功能性(functional)。其中,前者几乎是从Java第一版就存在了,而功能性始于2014年3月推出的Java 1.8。

1.面向对象和功能性方法

作为一种经典的面向对象语言,Java允许用户创建灵活的对象结构。在Java 1.8出现了功能性特性之后,它不仅可以使用对象或方法,还可以使用lambdas(其本身是可执行代码,https://dzone.com/articles/java-lambda-method-reference)进行各项操作。而在功能性的世界中,您可以像在OO的世界中使用对象那样,去操作各种功能。

图片

2.使用OO方法重构代码

通过使用继承或组合的方式来处理各种接口和类,您可以创建出各种可重用的通用方案,从而减少程序的代码量,并提高可读性。如果一个类满足了如下的条件,那么它便可以在相同的公共结构中进行联合:

  • 具有相似的字段,并被识别为同一实体
  • 有父/子(parent/children)关系
  • 有相似目的的方法

3.使用功能性方法重构代码

与OO方法不同,这种方法提取具有相同行为的代码。例如,我们可以在如下两个示例中识别出相似之处:

  • 在具体实现上能够返回相同类型
  • 在具体实现上有相同的功能

4.使用两种方法进行重构的示例

假设我们有一个小型应用程序,其功能向正式员工(Employee)和合同工(Contractor)支付工资。每次完成支付工资后,我们都会打印一份Employee报告,并以不同的格式显示(https://dzone.com/articles/so-much-data-so-many-formats-a-conversion-service),即:正式员工为JSON格式,而合同工为XML格式。以下便是使用两种方法重构的示例:

图片

现在让我们来看一下其默认结构:

图片

5.重构类的结构

很明显,Contractor可以成为Employee类的子类。同时,makePayment可以被覆盖掉。当然,我们也可以创建一个Payable接口和提取makePayment的方法,不过让我们在此保持简单化。如下代码段所示,在重构之后,我们产生了一些共同的字段,以及可重用的构造函数。

图片

6.重构功能性

现在我们可以从功能性的角度,来回顾和发掘源代码中的相似之处:

图片

如上图所示,从打印报告中可以看出,我们可以使用相同的方式来进行处理,即:传递一个对象,并返回一个字符串。因此,我们可以将代码部分提取为一个可重用的功能,并将其动态地用于该业务的逻辑上。我们甚至可以将其拿到该业务的外部进行使用。

7.创建功能接口

为了判定正确的功能性接口(如,Predicate、Consumer、Function等),我们需要检查自己的输入和输出。在本例中,我们得到的是一个Object,并需要将其转换为String。

图片

该接口是由功能函数提供的。为了更加便于理解,我们用serialize方法创建一个自有的Converter接口。其对应的代码如下,具有极强的可读性:

图片

8.创建Lambda转换器(Converter)功能性接口

下面,我们可以在功能性接口的基础上,创建两个转换器:JSON和XML。它们都会去匹配已定义的签名,即:对象输入(Object Input)和字符串输出(String Output)。

图片

接着,让我们在代码中使用它们:

图片

9.在Employee类中封装转换器

与前面的方法类似,我们可以将此功能封装在Employee父类中,并在内部功能函数中使用它们。下图展示了如何在Employee类中封装转换器:

图片

10.审查最终版本

最后,我们初始化两个employee类,并遍历它们的支付执行情况和打印方法。

图片

我们将最终得到:

  • Employee父类中的Commons字段和方法被重用到了Contractor中
  • 可以在无需更改Employee类的情况下,提取功能函数转换器(我们可以在未来再做补充)
  • 提取的功能函数可以在Employee类之外被重用

11.点评    

总的说来,上述示例并不完美,且有待改进。例如,我们可以将Employee与Contractors类隐藏在接口的后面。您也可以试着去写一个简单的例子,以便只展示一些面对对象和功能特性。

当然,从严格意义上说,我创建的功能可能并非纯功能。而一些开发人员往往坚持认为:在Java中只有纯功能才是更好的。在此,我持保留意见。

12.小结    

让我们对上述内容小结一下:

使用面向对象的方法,我们可以将性质相似(similar-by-nature)的对象统一到同一个结构中。

使用功能性方法,我们则可以统一功能相似(similar-by-functionality)的代码。

这两种方法都能够让程序代码的可读性和可维护性得到显著提高。

原文链接:https://dzone.com/articles/refactoring-java-application-object-oriented-and-f

译者介绍

陈峻 (Julian Chen),51CTO社区编辑,具有十多年的IT项目实施经验,善于对内外部资源与风险实施管控,专注传播网络与信息安全知识与经验;持续以博文、专题和译文等形式,分享前沿技术与新知;经常以线上、线下等方式,开展信息安全类培训与授课。

责任编辑:武晓燕 来源: 51CTO技术栈
相关推荐

2018-12-12 08:15:13

物联网设备物联网IOT

2011-06-02 09:47:11

C语言重构

2022-08-23 09:00:00

Web测试工具自动化

2021-12-29 21:15:08

软件测试软件开发

2012-05-31 10:24:59

架构

2023-08-08 12:32:34

AMD公共云工程

2012-12-13 11:05:42

IBMdW

2010-07-23 10:54:09

优化SQL Serve

2010-07-20 09:33:14

Perl方法

2017-12-04 10:51:06

VMwareAWS

2011-06-28 09:53:43

iPhone诺基亚N9

2020-06-09 09:13:12

JavaScript重构对象

2013-06-24 10:21:47

面向对象Web应用JavaScript

2009-06-03 18:33:35

英特尔服务器Nehalem

2009-08-13 18:00:48

Eclipse重构功能扩展点

2009-12-18 11:23:30

Visual Web

2014-11-21 09:17:14

2021-07-12 15:17:08

机器学习化学元素氧化态

2020-05-08 10:29:49

CIO敏捷低代码

2023-01-10 09:38:09

面向对象系统
点赞
收藏

51CTO技术栈公众号