译者 | 陈峻,审校 | 孙淑娟,在重构方面,Java主要有两种主要方法,即:面向对象(object-oriented)和功能性(functional)。其中,前者几乎是从Java第一版就存在了,而功能性始于2014年3月推出的Java 1.8。,作为一种经典的面向对象语言,Java允许用户创建灵活的对象结构。在Java 1.8出现了功能性特性之后,它不仅可以使用对象或方法,还可以使用lambdas(其本身是可执行代码,https://dzone.com/articles/java-lambda-method-reference)进行各项操作。而在功能性的世界中,您可以像在OO的世界中使用对象那样,去操作各种功能。,通过使用继承或组合的方式来处理各种接口和类,您可以创建出各种可重用的通用方案,从而减少程序的代码量,并提高可读性。如果一个类满足了如下的条件,那么它便可以在相同的公共结构中进行联合:,与OO方法不同,这种方法提取具有相同行为的代码。例如,我们可以在如下两个示例中识别出相似之处:,假设我们有一个小型应用程序,其功能向正式员工(Employee)和合同工(Contractor)支付工资。每次完成支付工资后,我们都会打印一份Employee报告,并以不同的格式显示(https://dzone.com/articles/so-much-data-so-many-formats-a-conversion-service),即:正式员工为JSON格式,而合同工为XML格式。以下便是使用两种方法重构的示例:,现在让我们来看一下其默认结构:,很明显,Contractor可以成为Employee类的子类。同时,makePayment可以被覆盖掉。当然,我们也可以创建一个Payable接口和提取makePayment的方法,不过让我们在此保持简单化。如下代码段所示,在重构之后,我们产生了一些共同的字段,以及可重用的构造函数。,现在我们可以从功能性的角度,来回顾和发掘源代码中的相似之处:,如上图所示,从打印报告中可以看出,我们可以使用相同的方式来进行处理,即:传递一个对象,并返回一个字符串。因此,我们可以将代码部分提取为一个可重用的功能,并将其动态地用于该业务的逻辑上。我们甚至可以将其拿到该业务的外部进行使用。,为了判定正确的功能性接口(如,Predicate、Consumer、Function等),我们需要检查自己的输入和输出。在本例中,我们得到的是一个Object,并需要将其转换为String。,该接口是由功能函数提供的。为了更加便于理解,我们用serialize方法创建一个自有的Converter接口。其对应的代码如下,具有极强的可读性:,下面,我们可以在功能性接口的基础上,创建两个转换器:JSON和XML。它们都会去匹配已定义的签名,即:对象输入(Object Input)和字符串输出(String Output)。,接着,让我们在代码中使用它们:,与前面的方法类似,我们可以将此功能封装在Employee父类中,并在内部功能函数中使用它们。下图展示了如何在Employee类中封装转换器:,最后,我们初始化两个employee类,并遍历它们的支付执行情况和打印方法。,我们将最终得到:,总的说来,上述示例并不完美,且有待改进。例如,我们可以将Employee与Contractors类隐藏在接口的后面。您也可以试着去写一个简单的例子,以便只展示一些面对对象和功能特性。,当然,从严格意义上说,我创建的功能可能并非纯功能。而一些开发人员往往坚持认为:在Java中只有纯功能才是更好的。在此,我持保留意见。,让我们对上述内容小结一下:,使用面向对象的方法,我们可以将性质相似(similar-by-nature)的对象统一到同一个结构中。,使用功能性方法,我们则可以统一功能相似(similar-by-functionality)的代码。,这两种方法都能够让程序代码的可读性和可维护性得到显著提高。,原文链接:https://dzone.com/articles/refactoring-java-application-object-oriented-and-f,陈峻 (Julian Chen),51CTO社区编辑,具有十多年的IT项目实施经验,善于对内外部资源与风险实施管控,专注传播网络与信息安全知识与经验;持续以博文、专题和译文等形式,分享前沿技术与新知;经常以线上、线下等方式,开展信息安全类培训与授课。,
© 版权声明
文章版权归作者所有,未经允许请勿转载。