用多态和组合替换多个条件

这是一种众所周知的重构模式,可以将条件条件替换为多态性。 如果您不熟悉该模式,可以在此处查看 。 但是,一旦该类中有多个条件检查所基于的字段,该基本解决方案便会开始崩溃。 我们将研究一些有关如何使用这些可能性的想法。

有很多方法可以解决,因此我们将从最简单到最困难的工作,始终使用简单的示例来尽可能减少混乱。 那么,最简​​单的情况是什么? 看一看:

public class ClassWithConditionals
{private boolean conditional1;private EnumeratedType conditional2;public ClassWithConditionals(boolean cond1, EnumeratedType cond2){conditional1 = cond1;conditional2 = cond2;}public void method1(){if(conditional1){//do something}else{//do something else}}public void method2(){switch(conditional2){case CASE1://do somethingbreak;case CASE2://do something elsebreak;case CASE3://do something entirely differentbreak;}}
}enum EnumeratedType
{CASE1,CASE2,CASE3
}

因此,在此示例中,我们有ClassWithConditionals在其方法中使用的两个不同字段。 在一个合适的示例中,您将假设不仅使用给定的两个方法,还可以使用更多的方法,但是对于该示例,我们仅需要两个方法。 如果每个条件只使用一种方法,那么您就不必担心,因为维护成本仍然很低。 但是,只要执行这种条件检查的方法数量增加,就应该考虑进行这种重构。

通常,如果您要遵循用多态替换条件,您将最终得到六个类来解决此问题: booleanenum每个组合都有一个。 取而代之的是,我们将使用合成。

那么,第一步是什么? 首先,我们可能应该使用enum类型。 enum可以有自己的方法,可以以允许其根据特定enum做不同事情的方式定义这些方法。 因此,让我们将enum eratedType更改如下:

enum EnumeratedType
{CASE1(){public void doSomething(){//do something}},CASE2(){public void doSomething(){//do something else}},CASE3(){public void doSomething(){//do something entirely different}};public abstract void doSomething();
}

现在, method2仅需要将自身委托给conditional2.doSomething()

现在让我们修复boolean 。 我们创建一个接口,该接口对所有非封装类(为了测试起见,可能是包)都私有,称为Conditional1 。 然后我们用TrueFalse对其进行子类化。 这是代码:

interface Conditional1
{static Conditional1 TRUE = new True();static Conditional1 FALSE = new False();void doSomething();
}class True implements Conditional1
{public void doSomething(){//do something}
}class False implements Conditional1
{public void doSomething(){//do something else}
}

我决定在接口上创建TRUEFALSE实例的原因很简单:它们都是无状态类,这意味着每个实例都具有多个实例是没有意义的。 它还允许我们像enum一样调用它们。

同样,现在主要类仅需要委托。 这是固定类现在的样子

public class ClassWithConditionals
{public static ClassWithConditionals with(boolean cond1, EnumeratedType cond2){Conditional1 conditional1;if(cond1)conditional1 = Conditional1.TRUE;elseconditional1 = Conditional1.FALSE;return new ClassWithConditionals(conditional1, cond2);}private Conditional1 conditional1;private EnumeratedType conditional2;ClassWithConditionals(Conditional1 cond1, EnumeratedType cond2){this.conditional1 = cond1;this.conditional2 = cond2;}public void method1(){conditional1.doSomething();}public void method2(){conditional2.doSomething();}
}

这里有些奇怪。 我们已经用另一个替换了一个条件。 我们的构造函数足以接受一个Conditional1 ,但是我们有一个静态工厂方法,该方法仍然采用boolean并对此进行条件检查。

考虑到从技术上讲,除非有多种方法进行检查,否则我们不会重构此代码,因此,我们进行了许多检查并将其归为一类。 同样,在工厂中通常认为有条件的还可以,将所有检查强制到一个位置,并允许多态性从那里接管。 您不必使用静态工厂方法作为工厂,但是它是最快速,最简单的实时设置方法。 允许调用新ClassWithConditionals对象的创建代码的代码仍然可以按过去的方式传递boolean的另一个好处是,它允许我们封装和隐藏基于条件的类的实现细节。 新ClassWithConditionals创建者无需担心创建Conditional1对象,甚至无需知道它的存在。

我们仍然希望构造函数接受Conditional1对象,这有两个原因:1)它将条件逻​​辑保存在工厂中,而不是构造函数中,后者是首选,并且2)它允许我们传递Conditional1对象的测试双精度。

实际上,由于第2点,我们应该经常考虑将其enum转换为更类似于Conditional1的静态实例。 这将允许您更多地使用测试双打。 它还将有助于继承或通过组合进行扩展,这将在稍后进行讨论。

可以想到很多小变化。 首先,条件boolean不需要booleanenum 。 可以有一组基于数字或其他条件的条件表达式。 通常,在这些情况下,我们用一个小的辅助方法来代替支票以使其更清晰,即if(numberOfPeople <= 3)...变成if(isACrowd(numberOfPeople))... 我们可以更进一步,并创建通过工厂创建的GroupsOfPeople层次结构。 如果工厂的值为1,则返回SinglePerson ; 给定2,则返回Company对象; 给定3或更多,它将返回一个Crowd对象。 这些对象中的每个对象都有其自己的方法,这样可以帮助减少原始类中的代码量。

另一个变化是当不同的条件字段集分层在一起时( if(condition1 && condition2)等)。 为了解决这个问题,您可以走继承路线并创建爆炸的类以覆盖所有组合。 另一个选择是用一个小的层次结构替换一个条件对象,该层次结构接受委托方法中的其他条件对象,在该方法中,它仍然具有一些条件代码,但是可读性更强。 例如,您可以将使用两个布尔值的类转换为如下形式:

public class ClassWithConditionals
{public static ClassWithConditionals with(boolean condition1, boolean condition2){Conditional1 cond1;if(condition1)cond1 = Conditional1.TRUE;elsecond1 = Conditional1.FALSE;return new ClassWithConditionals(cond1, condition2);}private Conditional1 condition1;private boolean condition2;ClassWithConditionals(Conditional1 condition1, boolean condition2){this.condition1 = condition1;this.condition2 = condition2;}public void method(){condition1.method(condition2);}
}interface Conditional1
{static Conditional1 TRUE = new True();static Conditional1 FALSE = new False();void method(boolean condition2);
}class True implements Conditional1
{public void method(boolean condition2){if(condition2){//do something}else{//do something else}}
}class False implements Conditional1
{public void method(boolean condition2){if(!condition2){//do something really different}//and do this}
}

Condition1method接受一个布尔值,然后使用它进行更多的条件处理。

另外,如果所有逻辑都允许,则可以创建一组类来替换其中一个条件,然后让其创建代码接受其他条件以决定其创建的一部分。 例如:

public class ClassWithConditionals
{public static ClassWithConditionals from(boolean condition1, boolean condition2){return new ClassWithConditionals(Conditional1.from(condition1, condition2));}private Conditional1 conditionOne;ClassWithConditionals(Conditional1 conditionOne){this.conditionOne = conditionOne;}public int method(){return conditionOne.method() * -6;}
}interface Conditional1
{static Conditional1 from(boolean condition1, boolean condition2){if(condition1)return True.with(condition2);elsereturn False.with(condition2);}int method();
}class True implements Conditional1
{public static True with(boolean condition2){if(condition2)return new True(5);elsereturn new True(13);}private int secondary;public True(int secondary){this.secondary = secondary;}public int method(){return 2 * secondary;}
}class False implements Conditional1
{public static False with(boolean condition2){if(condition2)return new False((x, y) -> x - y, 31);elsereturn new False((x, y) -> x * y, 61);}private final BinaryOperator operation;private final int secondary;public False(BinaryOperator operation, int secondary){this.operation = operation;this.secondary = secondary;}public int method(){return operation.apply(4, secondary);}
}

对于True ,第二个条件决定method计算中的次要数字。 在False ,它会执行此操作并找出要应用于计算的运算符。

我不确定是否会发生类似的事情,但是如果确实发生了,您现在知道了一种解决方法。

总的来说,这整套重构从本质上将代码从单个类更改为Facade。 它需要大量的新类,并且使您可以使用与以前的单个类几乎完全相同的方式来使用整个工具包和kaboodle,唯一真正的区别是调用静态工厂方法而不是构造函数。

这不是特别重要; 我只是想向您指出。

希望您不必担心继承或“通过合成扩展”此类。 但是您可能必须这样做。

如果您要编写的扩展仅真正改变了条件对象的功能,则可以简单地编写一个新的Factory,为构造函数提供一组新的条件对象。 例如,您可以将此静态工厂方法添加到ClassWithConditionals的最新版本中:

public static ClassWithConditionals different(int value)
{return new ClassWithConditionals(new SimpleConditional1(value));
}

SimpleConditional1看起来像这样

class SimpleConditional1 implements Conditional1
{private final int value;public SimpleConditional1(int value){this.value = value;}public int method(){return value;}
}

除此之外,您还可以提供原始需要的任何条件对象,并覆盖您需要覆盖的所有方法。

因此,这就是我用一个更多的OO选项替换多个条件的结果。 您还有其他方法可以做到吗? 您是否有一个无法奏效的示例,您希望我对此加以抨击? 让我知道,我会看看可以做什么。

谢谢阅读。

翻译自: https://www.javacodegeeks.com/2015/01/replacing-multiple-conditionals-with-polymorphism-and-composition.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/361099.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

canvas画饼图

<style> body { background: black; text-align: center; } #cans { background: white; } </style> <script> function disToRad(n){//将度数表示弧度计算的方法 return n*Math.PI/180;//π用PI表示&#xff0c;π180&#xff0c;所以1PI/180 } w…

Web设计趋势分析

本文译自网站设计公司weavora.com&#xff0c;介绍了在他们眼里 8 个 Web 设计趋势&#xff1a;单页面、用照片做背景、色块设计、超大号的图片、聚焦简洁、响应式设计、视差滚动、强调字体&#xff0c;每个趋势后面都附了数个案例&#xff0c;相信对网站设计师会有一定的参考价…

使用Jasmine,Spock和Nashorn测试JVM服务器端JavaScript

JavaScript使用不仅限于浏览器中的客户端代码或NodeJS支持的服务器端代码。 许多基于JVM的项目都将其用作内部脚本语言。 测试这种功能既不简单也不标准。 在本文中&#xff0c;我打算演示一种使用成熟的工具&#xff08;例如Jasmine &#xff0c; Spock和Nashorn在服务器端JVM…

C#中的多态

封装、继承、多态&#xff0c;面向对象的三大特性&#xff0c;前两项理解相对容易&#xff0c;但要理解多态&#xff0c;特别是深入的了解&#xff0c;对于初学者而言可能就会有一定困难了。我一直认为学习OO的最好方法就是结合实践&#xff0c;封装、继承在实际工作中的应用随…

AJAX JSON

1、AJAX [1] AJAX简介 > 全称&#xff1a; Asynchronous JavaScript And XML > 异步的JavaScript和XML > AJAX就是通过JavaScript向服务器发送请求&#xff0c;并接收响应&#xff0c;然后我们在通过DOM来修改页面。 > XML指的是服务器响应的…

在WildFly 8.2中修补焊接3 – Java EE 8的第一个实验RI

Java EE 8一直在发展&#xff0c;并且已经提出了几个新的组件JSR。 JSR 365将定义CDI 2.0的规范。 红帽公司已经开始研究Weld 3的实现原型&#xff0c;并且Alpha3最近发布了 。 Red Hat的Java EE 8兼容应用服务器将是WildFly&#xff0c;将在其中实现所有不同的技术。 同时&am…

mat-form-field must contain a MatFormFieldControl错误的解决方法

下面的代码竟然出错了&#xff1a; <mat-form-field><input matInput placeholder"输入名称"></mat-form-field> 错误提示的莫名其妙&#xff0c;其实只要导入以下模块就可了&#xff1a; imports: [MatFormFieldModule,MatInputModule,] 更多专业…

lua# lua5.1.4 源码文件作用一览

写了个脚本列出lua源码C文件头部的注释&#xff0c;作为我有一搭没一搭以Lua为对象学习编译原理的开端。 lua5.1.4全部的源码有35个C文件&#xff0c;17216行代码。每个文件基本的功能如下 ./output_lua_sources_comments.sh ~/resources/sources/lua/src …

带有Hibernate OGM的NoSQL –第一部分:持久化您的第一个实体

Hibernate OGM的第一个最终版本已经发布 &#xff0c;团队从发布狂潮中恢复了一些。 因此&#xff0c;他们考虑建立一系列教程式博客&#xff0c;使您有机会轻松地从Hibernate OGM重新开始。 感谢Gunnar Morling&#xff08; gunnarmorling &#xff09;创建了本教程。 介绍 不…

为自己写程序之JavsScript代码段测试器

JavaScript的测试&#xff0c;通常是在Firefox的firebug插件中测试的。不过有时只是测试几行代码都要写一个html&#xff0c;再打开浏览器测试运行结果&#xff0c;感觉并不是很方便。 今天花了点时间做了一个简易的JS片段测试器。其实这主要是看了IronJs开源项目以后&#xff…

sizeof和strlen的区别(其中涉及NUL的讲解)

本文是自己结合平时所学的知识&#xff0c;对sizeof和strlen的区别进行了总结&#xff0c;如有不对的地方还请批评指证&#xff0c;共同进步&#xff01;&#xff01;&#xff01; 一、从C语言的定义上来讲 1、sizeof是关键字&#xff0c;而strlen是包含在string.h头文件中的一…

我的Wiki:使用JConsole对WildFly(或JBoss AS7)进行远程JMX访问

与以前的版本相比&#xff0c;JBoss AS7的目标之一是使其在默认情况下更加安全。 受此目标直接影响的领域之一是&#xff0c;您不再期望服务器在端口上公开某些服务&#xff0c;而无需任何身份验证/授权就可以访问它。 请记住&#xff0c;在以前的JBoss AS版本中&#xff0c;只…

js判断对象类型

1.typeof typeof只能判断区分基本类型&#xff0c;number、string、boolean、undefined和object,function&#xff1b; typeof 0; //number; typeof true; //boolean; typeof undefined; //undefined; typeof "hello world" //string; typeof function(){}; …

我喜欢的类型

http://v.qq.com/cover/h/hfd581s2y9unvy8.html?vidp0011ocge8q 转载于:https://www.cnblogs.com/sliz/archive/2012/12/09/2809742.html

使用WildFly和Java EE 7映像与Docker提供者一起流浪

什么是无业游民&#xff1f; Vagrant是创建虚拟开发环境的简化且可移植的方式。 它可与多种虚拟化软件一起使用&#xff0c;例如VirtualBox&#xff0c;VMWare&#xff0c;AWS等。 它还可以与多种配置软件一起使用&#xff0c;例如Ansible&#xff0c;Chef&#xff0c;Puppet或…

Apache Nutch 1.6 发布

Apache Nutch 1.6 发布&#xff0c;该版本修复了超过 20 个 bug&#xff0c;新功能包括&#xff1a;新的 HostNormalizer&#xff0c;可通过 MIME-type 和 Indexer API 的功能增强来动态设置 fetchInterval &#xff0c;更新 Tika 到 1.2 版本&#xff0c;更新 Autimaton 到 1.…

EE Servlet 3:如何在Web应用程序中设置后端服务

在Web应用程序中&#xff0c;提供用户界面&#xff08;UI&#xff09;通常只是工作的一半。 许多应用程序都有后端服务支持的要求。 后端服务的一些示例是调度程序进程&#xff08;批处理&#xff09;&#xff0c;侦听队列并在消息进入时作出响应&#xff0c;或者是简单的事情&…

es6解构赋值

解构赋值语法是一个 Javascript 表达式,这使得可以将值从数组或属性从对象提取到不同的变量中。 数组解构赋值&#xff1a; {let a,b,rest;[a,b][1,2];console.log(a,b); //1 2 } {let a,b,rest;[a,b,...rest][1,2,3,4,5]; console.log(a,b,rest); //1 2 [ 3, 4, 5 ]…

Jquery插件之ajaxForm

如今ajax满天飞&#xff0c;作为重点的form自然也受到照顾。 其实&#xff0c;我们在平常使用Jquery异步提交表单&#xff0c;一般是在submit()中&#xff0c;使用$.ajax进行。比如&#xff1a; $(function(){$(#myForm).submit(function(){$.ajax({url:"/WebTest/test/te…

休眠锁定模式–乐观锁定模式如何工作

显式乐观锁定 在上一篇文章中 &#xff0c;我介绍了Java持久性锁定的基本概念。 隐式锁定机制可防止丢失更新 &#xff0c;它适用于我们可以主动修改的实体。 虽然隐式乐观锁定是一种广泛使用的技术&#xff0c;但是很少有人了解显式乐观锁定模式的内部工作原理。 当锁定的实…