js 使用多态替换条件语句_用多态和组成替换多个条件

js 使用多态替换条件语句

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

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

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

js 使用多态替换条件语句

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

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

相关文章

python连不上树莓派_树莓派respberry中cmake编译链接python2.7库不成功,求教

本人在树莓派pi3平台上&#xff0c;下载respberry 5.4 lite 编译一个github上的项目&#xff0c;该项目用cmake管理&#xff0c;目前看到 链接库阶段不成功。请教如何调整cmake的链接库目录&#xff0c;或是怎样直接调整生成的makefile&#xff0c;用于连编是一个主动降噪的项目…

java8 camel_WildFly 8的Camel子系统集成了Java EE –入门

java8 camel就在三天前&#xff0c;围绕Thomas Diesler&#xff08; tdiesler &#xff09;的团队发布了WildFly-Camel子系统的2.0.0.CR1版本&#xff0c;它允许您将Camel Routes添加为WildFly配置的一部分。 路由可以部署为JavaEE应用程序的一部分。 JavaEE组件可以访问Camel …

php简单网站源码包含数组_PHP无限分级阶梯递归数组排列 以及多层嵌套数组在网页中的解析源码...

数组递归PHP无限分级 将数组递归转化成阶梯型数组 数组中包含数组 使用更方便 查看也很方便&#xff0c;网上找的都不是很好&#xff0c;自己写的才是王道简单的递归 没有什么特别之处 只是自己写的容易理解一点而已&#xff01;代码在下面$array[0]array(fid>1,top>0,n…

工程师总结:单片机C语言编程心得

一、.H文件与.C文件的关系&#xff1a;迄今为止&#xff0c;写过的程序都是一些很简单的程序&#xff0c;从来没有想到要自己写.H文件&#xff0c;也不知道.H文件到底什么用&#xff0c;与.C文件什么关系。只是最近写键盘程序&#xff0c;参考别人的程序时&#xff0c;发现别人…

在建工地扬尘在线监控系统推荐_vocs在线监控系统安装的法规依据

voc是挥发性有机化合物的简称&#xff0c;在环保监测领域指活泼累得挥发性有机物&#xff0c;具有特殊的刺激性气味&#xff0c;vocs在线监控系统是专门用于监测voc的排放浓度及排放量的。当一个空间内的voc浓度达到一定数值&#xff0c;人们会出现头晕、恶心、呕吐等多种症状&…

jboss默认进程名称_快速指南:剖析JBoss BPM跨进程通信

jboss默认进程名称&#xff08;文章来宾与北美红帽公司高级解决方案架构师杰伊保拉杰合着&#xff09; 几周的提示与技巧文章将深入探讨JBoss BPM Suite&#xff0c;特别是有关如何在两个流程之间进行通信的问题。 在深入了解解决方案细节之前&#xff0c;让我们首先约束将要讨…

嵌入式C语言编程——.h文件与.c文件

.h文件与.c文件的关系参考高手的程序时&#xff0c;发现别人写的严格的程序都带有一个“KEY.H”&#xff0c;里面定义了.C文件里用到的自己写的函数&#xff0c;如Keyhit()、Keyscan()等。.H文件就是头文件&#xff0c;估计就是Head的意思吧&#xff0c;这是规范程序结构化设计…

sql between包括两端吗_SQL简单查询

1.SQL简单查询知识点概要&#xff1a;1. 1基本的查询语句&#xff1a;1.1 .1 Select<列明1>,<列明2>1.1.2 From<表名>1.2. SQL运行数据&#xff1a;1.2.1 SQL语句以英文分号(;)结尾。1.2.2列明用英文的单引号&#xff0c;不区分英文的大小写。1.2.3 列明命名…

使用javac编译java源文件_Java种动态加载(一)——java源文件动态编译为class文件...

Java类动态加载(一)——java源文件动态编译为class文件最近在做java动态加载这方面的工作&#xff0c;起初也遇到了很多困难。网上关于这方便的东西很零散&#xff0c;为了便于日后回过头来再看&#xff0c;于是我将这几天的心得体会总结如下。什么情况下会需要用java程序动态的…

gradle构建多模块项目_Gradle入门:创建多项目构建

gradle构建多模块项目尽管我们可以仅使用一个模块来创建一个运行中的应用程序&#xff0c;但有时将我们的应用程序划分为多个较小的模块是比较明智​​的。 因为这是一个相当普遍的用例&#xff0c;所以每个自重的构建工具都必须支持它&#xff0c;Gradle也不例外。 如果Gradl…

python第一条入门程序_Python语言函数代码的执行流程

https://www.xin3721.com/eschool/pythonxin3721/Python语言函数代码的执行流程&#xff0c;为了保证函数的定义先于其首次调用时执行&#xff0c;我们需要知道中语句的执行顺序。执行总是从程序的第一行代码开始的&#xff0c;从上到下&#xff0c;从左到右&#xff0c;按顺序…

嵌入式C中,全局变量滥用的后果竟如此严重?

说起全局变量&#xff0c;就不得不提到“全局变量&#xff0c;局部变量&#xff0c;静态全局变量&#xff0c;静态局部变量”&#xff0c;这些都是编程语言中的基本概念。变量分为局部与全局&#xff0c;局部变量又可称之为内部变量。由某对象或某个函数所创建的变量通常都是局…

php登陆页面修改密码的功能,使用bootstrap创建登录注册页面并实现表单验证功能...

本篇文章给大家介绍一下使用bootstrap创建登录注册页面并实现单验证功能的方法。有一定的参考价值&#xff0c;有需要的朋友可以参考一下&#xff0c;希望对大家有所帮助。用bootstrap做登入注册页面&#xff0c;使用validate做表单验证技术&#xff1a;bootstrap&#xff0c;f…

java压缩文件读取_用Java读取/写入压缩和非压缩文件

java压缩文件读取这篇文章的主要原因是尝试不要重复自己&#xff08; DRY &#xff09;&#xff0c;因为通常&#xff0c;我会遇到递归的需求&#xff0c;即读写压缩的和非压缩的文件&#xff08;主要是JSON和CSV&#xff09;。 首先让我们看看如何读取文本文件。 注意我正在使…

vue router 参数_Vue.js项目开发技术解析

Vue.js项目开发技术解析一、Vue.js实例在一个Vue.js工程中&#xff0c;用于显示内容最基层的实例称之为根实例。通过该实例可以进行页面或组件的更新和显示。对于项目本身而言&#xff0c;无论是什么样的页面&#xff0c;都要基于该根实例进行显示。1.1、何为构造器对于Vue.js项…

deepin nginx连接php,利用docker运行nginx加上本机的php-fpm。访问html文件正常,但是访问php文件就报错404...

最近在虚拟机弄了一个deepin系统。打算使用docker加载nginx/1.17.6加在官方下载的PHP7.4.1(这个是本地编译的)。出现了以下的问题&#xff1a;1)访问html文件正常&#xff0c;但是访问php文件报404。根据网上的文章尝试&#xff1a;1)php运行不成功。在PHP的cli模式下是正常的。…

C语言中面向对象编程

C语言中面相对象的编程面向对象的重要思想就是数据隐藏&#xff0c;在面向对象语言中&#xff0c;对象可以包含私有变量。这样我们可以说他们具有内部状态,这些内部状态对其他对象是透明的。全局变量可以通过设置变量作用域来模拟私有变量&#xff08;甚至友元对象&#xff09;…

jmeter测试客户端_如何在JMeter中执行客户端Web性能测试?

jmeter测试客户端在本文中&#xff0c;我们将看到如何使用Jmeter插件进行客户端性能测试。 我将使用jmeter webdriver插件。 在开始本主题之前&#xff0c;请阅读我以前的文章中有关客户端性能测试的一些基本信息。 因此&#xff0c;让我们开始吧&#xff1a; 安装 通过这篇文…

java编译程序的基本命令是什么,【填空题】Java中编译java 程序的命令是 1 ,执行java程序的命令是 java 。...

【填空题】Java中编译java 程序的命令是 1 ,执行java程序的命令是 java 。更多相关问题[多选] 下列骨折中属于关节外骨折的是()。[多选] 骨折早期局部的并发症包括()。[多选] 典型中1/3段锁骨骨折的内侧端移位方向是()。[多选] 骨关节结核的诊断要点包括()。[多选] 骨痂形成的方…

python range 小数_python中如何表示一个无限循环小数?(不用分数的形式)python,使用range语...

python中如何表示一个无限循环小数&#xff1f;(不用分数的形式) python,使用range语www.zhiqu.org 时间&#xff1a; 2020-12-07只能用分数或者你自己设计一个对象&#xff0c;保留指定长度的有效位range()只能产生int无法产生float但是可以曲线救国for i in range(0,10,1…