范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文
国学影视

java开发精准优化ifelse,干掉,过多,烂代码

  大家好,
  今天,跟大家聊一聊, 现实中的代码往往存在着过多的 if...else。虽然 if...else 是必须的,但滥用 if...else 会对代码的可读性、可维护性造成很大伤害,进而危害到整个软件系统。
  现在软件开发领域出现了很多新技术、新概念,但 if...else 这种基本的程序形式并没有发生太大的变化。使用好 if...else 不仅对于现在,而且对于将来,都是十分有意义的。
  今天我们就来看看如何"干掉"代码中的 if...else,还代码以清爽。  问题一:if…else 过多问题表现
  if...else 过多的代码可以抽象为下面这段代码。其中只列出5个逻辑分支,但实际工作中,能见到一个方法包含10个、20个甚至更多的逻辑分支的情况。另外,if...else 过多通常会伴随着另两个问题:逻辑表达式复杂和 if...else 嵌套过深。 if (condition1) {  } else if (condition2) {  } else if (condition3) {  } else if (condition4) {  } else {  }
  通常,if...else 过多的方法,通常可读性和可扩展性都不好。从软件设计角度讲,代码中存在过多的 if...else 往往意味着这段代码违反了违反单一职责原则和开闭原则。
  因为在实际的项目中,需求往往是不断变化的,新需求也层出不穷。所以,软件系统的扩展性是非常重要的。
  而解决 if...else 过多问题的最大意义,往往就在于提高代码的可扩展性。 如何解决
  接下来我们来看如何解决 if...else 过多的问题。下面我列出了一些解决方法。 表驱动 职责链模式 注解驱动 事件驱动 有限状态机 Optional Assert 多态 方法一:表驱动 介绍
  对于逻辑表达模式固定的 if...else 代码,可以通过某种映射关系,将逻辑表达式用表格的方式表示;再使用表格查找的方式,找到某个输入所对应的处理函数,然后使用这个处理函数进行运算。 适用场景
  逻辑表达模式固定的 if...else 实现与示例if (param.equals(value1)) {     doAction1(someParams); } else if (param.equals(value2)) {     doAction2(someParams); } else if (param.equals(value3)) {     doAction3(someParams); } // ...
  可重构为 Map<?, Function<?> action> actionMappings = new HashMap<>(); // 这里泛型 ? 是为方便演示,实际可替换为你需要的类型  // When init actionMappings.put(value1, (someParams) -> { doAction1(someParams)}); actionMappings.put(value2, (someParams) -> { doAction2(someParams)}); actionMappings.put(value3, (someParams) -> { doAction3(someParams)});  // 省略 null 判断 actionMappings.get(param).apply(someParams);
  上面的示例使用了 Java 8 的 Lambda 和 Functional Interface,这里不做讲解。
  表的映射关系,可以采用集中的方式,也可以采用分散的方式, 即每个处理类自行注册。也可以通过配置文件的方式表达。总之,形式有很多。
  还有一些问题,其中的条件表达式并不像上例中的那样简单,但稍加变换,同样可以应用表驱动。下面借用《编程珠玑》中的一个税金计算的例子: if income <= 2200   tax = 0 else if income <= 2700   tax = 0.14 * (income - 2200) else if income <= 3200   tax = 70 + 0.15 * (income - 2700) else if income <= 3700   tax = 145 + 0.16 * (income - 3200) ...... else   tax = 53090 + 0.7 * (income - 102200)
  对于上面的代码,其实只需将税金的计算公式提取出来,将每一档的标准提取到一个表格,再加上一个循环即可。具体重构之后的代码不给出,大家自己思考。 方法二:职责链模式
  介绍:
  当 if...else 中的条件表达式灵活多变,无法将条件中的数据抽象为表格并用统一的方式进行判断时,这时应将对条件的判断权交给每个功能组件。并用链的形式将这些组件串联起来,形成完整的功能。 适用场景
  条件表达式灵活多变,没有统一的形式。 实现与示例
  职责链的模式在开源框架的 Filter、Interceptor 功能的实现中可以见到很多。下面看一下通用的使用模式:
  重构前: public void handle(request) {     if (handlerA.canHandle(request)) {         handlerA.handleRequest(request);     } else if (handlerB.canHandle(request)) {         handlerB.handleRequest(request);     } else if (handlerC.canHandle(request)) {         handlerC.handleRequest(request);     } }
  重构后: public void handle(request) {   handlerA.handleRequest(request); }  public abstract class Handler {   protected Handler next;   public abstract void handleRequest(Request request);   public void setNext(Handler next) { this.next = next; } }  public class HandlerA extends Handler {   public void handleRequest(Request request) {     if (canHandle(request)) doHandle(request);     else if (next != null) next.handleRequest(request);   } }
  当然,示例中的重构前的代码为了表达清楚,做了一些类和方法的抽取重构。现实中,更多的是平铺式的代码实现。 注:职责链的控制模式
  职责链模式在具体实现过程中,会有一些不同的形式。从链的调用控制角度看,可分为外部控制和内部控制两种。
  外部控制不灵活,但是减少了实现的难度。职责链上某一环上的具体实现不用考虑对下一环的调用,因为外部统一控制了。但是一般的外部控制也不能实现嵌套调用。如果有嵌套调用,并且希望由外部控制职责链的调用,实现起来会稍微复杂。具体可以参考 Spring Web Interceptor 机制的实现方法。
  内部控制就比较灵活,可以由具体的实现来决定是否需要调用链上的下一环。但如果调用控制模式是固定的,那这样的实现对于使用者来说是不便的。
  设计模式在具体使用中会有很多变种,大家需要灵活掌握
  方法三:注解驱动
  介绍
  通过 Java 注解(或其它语言的类似机制)定义执行某个方法的条件。在程序执行时,通过对比入参与注解中定义的条件是否匹配,再决定是否调用此方法。具体实现时,可以采用表驱动或职责链的方式实现。
  适用场景
  适合条件分支很多的,对程序扩展性和易用性均有较高要求的场景。通常是某个系统中经常遇到新需求的核心功能。
  实现与示例
  很多框架中都能看到这种模式的使用,比如常见的 Spring MVC。因为这些框架很常用,demo 随处可见,所以这里不再上具体的演示代码了。
  这个模式的重点在于实现。现有的框架都是用于实现某一特定领域的功能,例如 MVC。故业务系统如采用此模式需自行实现相关核心功能。主要会涉及反射、职责链等技术。具体的实现这里就不做演示了。
  方法四:事件驱动
  介绍
  通过关联不同的事件类型和对应的处理机制,来实现复杂的逻辑,同时达到解耦的目的。 适用场景
  从理论角度讲,事件驱动可以看做是表驱动的一种,但从实践角度讲,事件驱动和前面提到的表驱动有多处不同。具体来说:
  表驱动通常是一对一的关系;事件驱动通常是一对多; 表驱动中,触发和执行通常是强依赖;事件驱动中,触发和执行是弱依赖
  正是上述两者不同,导致了两者适用场景的不同。具体来说,事件驱动可用于如订单支付完成触发库存、物流、积分等功能。
  实现与示例
  实现方式上,单机的实践驱动可以使用 Guava、Spring 等框架实现。分布式的则一般通过各种消息队列方式实现。但是因为这里主要讨论的是消除 if...else,所以主要是面向单机问题域。因为涉及具体技术,所以此模式代码不做演示。 方法五:有限状态机介绍
  有限状态机通常被称为状态机(无限状态机这个概念可以忽略)。先引用维基百科上的定义:
  有限状态机(英语:finite-state machine,缩写:FSM),简称状态机,是表示有限个状态以及在这些状态之间的转移和动作等行为的数学模型。
  其实,状态机也可以看做是表驱动的一种,其实就是当前状态和事件两者组合与处理函数的一种对应关系。当然,处理成功之后还会有一个状态转移处理。
  适用场景
  虽然现在互联网后端服务都在强调无状态,但这并不意味着不能使用状态机这种设计。其实,在很多场景中,如协议栈、订单处理等功能中,状态机有这其天然的优势。因为这些场景中天然存在着状态和状态的流转。
  实现与示例
  实现状态机设计首先需要有相应的框架,这个框架需要实现至少一种状态机定义功能,以及对于的调用路由功能。状态机定义可以使用 DSL 或者注解的方式。原理不复杂,掌握了注解、反射等功能的同学应该可以很容易实现。
  参考技术:
  Apache Mina State Machine
  Apache Mina 框架,虽然在 IO 框架领域不及 Netty,但它却提供了一个状态机的功能。https://mina.apache.org/mina-project/userguide/ch14-state-machine/ch14-state-machine.html。有自己实现状态机功能的同学可以参考其源码。
  Spring State Machine
  Spring 子项目众多,其中有个不显山不露水的状态机框架 —— Spring State Machine https://projects.spring.io/spring-statemachine/。可以通过 DSL 和注解两种方式定义。
  上述框架只是起到一个参考的作用,如果涉及到具体项目,需要根据业务特点自行实现状态机的核心功能。
  方法六:Optional
  介绍
  Java 代码中的一部分 if...else 是由非空检查导致的。因此,降低这部分带来的 if...else 也就能降低整体的 if...else 的个数。
  Java 从 8 开始引入了 Optional 类,用于表示可能为空的对象。这个类提供了很多方法,用于相关的操作,可以用于消除 if...else。开源框架 Guava 和 Scala 语言也提供了类似的功能。
  使用场景
  有较多用于非空判断的 if...else。 实现与示例
  传统写法: String str = "Hello World!"; if (str != null) {     System.out.println(str); } else {     System.out.println("Null"); }
  使用 Optional 之后: 1 Optional strOptional = Optional.of("Hello World!"); 2 strOptional.ifPresentOrElse(System.out::println, () -> System.out.println("Null"));
  Optional 还有很多方法,这里不一一介绍了。但请注意,不要使用 get() 和 isPresent() 方法,否则和传统的 if...else 无异。
  扩展:Kotlin Null Safety
  Kotlin 带有一个被称为 Null Safety 的特性:
  bob?.department?.head?.name
  对于一个链式调用,在 Kotlin 语言中可以通过 ?. 避免空指针异常。如果某一环为 null,那整个链式表达式的值便为 null。
  方法七:Assert 模式介绍
  上一个方法适用于解决非空检查场景所导致的 if...else,类似的场景还有各种参数验证,比如还有字符串不为空等等。很多框架类库,例如 Spring、Apache Commons 都提供了工具里,用于实现这种通用的功能。这样大家就不必自行编写 if...else 了。
  Apache Commons Lang 中的 Validate 类:https://commons.apache.org/proper/commons-lang/javadocs/api-3.1/org/apache/commons/lang3/Validate.html
  Spring 的 Assert 类:https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/util/Assert.html  使用场景
  通常用于各种参数校验 扩展:Bean Validation
  类似上一个方法,介绍 Assert 模式顺便介绍一个有类似作用的技术 —— Bean Validation。Bean Validation 是 Java EE 规范中的一个。
  Bean Validation 通过在 Java Bean 上用注解的方式定义验证标准,然后通过框架统一进行验证。也可以起到了减少 if...else 的作用。
  方法八:多态
  介绍
  使用面向对象的多态,也可以起到消除 if...else 的作用。在代码重构这本书中,对此也有介绍:
  https://refactoring.com/catalog/replaceConditionalWithPolymorphism.html
  使用场景
  链接中给出的示例比较简单,无法体现适合使用多态消除 if...else 的具体场景。一般来说,当一个类中的多个方法都有类似于示例中的 if...else 判断,且条件相同,那就可以考虑使用多态的方式消除 if...else。
  同时,使用多态也不是彻底消除 if...else。而是将 if...else 合并转移到了对象的创建阶段。在创建阶段的 if..,我们可以使用前面介绍的方法处理。
  小结
  上面这节介绍了 if...else 过多所带来的问题,以及相应的解决方法。除了本节介绍的方法,还有一些其它的方法。比如,在《重构与模式》一书中就介绍了"用 Strategy 替换条件逻辑"、"用 State 替换状态改变条件语句"和"用 Command 替换条件调度程序"这三个方法。其中的"Command 模式",其思想同本文的"表驱动"方法大体一致。另两种方法,因为在《重构与模式》一书中已做详细讲解,这里就不再重复。
  何时使用何种方法,取决于面对的问题的类型。上面介绍的一些适用场景,只是一些建议,更多的需要开发人员自己的思考。 问题二:if…else 嵌套过深问题表现
  if...else 多通常并不是最严重的的问题。有的代码 if...else 不仅个数多,而且 if...else 之间嵌套的很深,也很复杂,导致代码可读性很差,自然也就难以维护。
  if (condition1) {     action1();     if (condition2) {         action2();         if (condition3) {             action3();             if (condition4) {                 action4();             }         }     } }
  if...else 嵌套过深会严重地影响代码的可读性。当然,也会有上一节提到的两个问题。 如何解决
  上一节介绍的方法也可用用来解决本节的问题,所以对于上面的方法,此节不做重复介绍。这一节重点一些方法,这些方法并不会降低 if...else 的个数,但是会提高代码的可读性: 抽取方法 卫语句
  方法一:抽取方法介绍
  抽取方法是代码重构的一种手段。定义很容易理解,就是将一段代码抽取出来,放入另一个单独定义的方法。借
  用 https://refactoring.com/catalog/extractMethod.html  中的定义: 适用场景
  if...else 嵌套严重的代码,通常可读性很差。故在进行大型重构前,需先进行小幅调整,提高其代码可读性。抽取方法便是最常用的一种调整手段。 实现与示例
  重构前:
  public void add(Object element) {   if (!readOnly) {     int newSize = size + 1;     if (newSize > elements.length) {       Object[] newElements = new Object[elements.length + 10];       for (int i = 0; i < size; i++) {         newElements[i] = elements[i];       }        elements = newElements     }     elements[size++] = element;   } }
  重构后: public void add(Object element) {   if (readOnly) {     return;   }    if (overCapacity()) {     grow();   }    addElement(element); }
  方法二:卫语句介绍
  在代码重构中,有一个方法被称为"使用卫语句替代嵌套条件语句"https://refactoring.com/catalog/replaceNestedConditionalWithGuardClauses.html 。直接看代码:
  double getPayAmount() {     double result;     if (_isDead) result = deadAmount();     else {         if (_isSeparated) result = separatedAmount();         else {             if (_isRetired) result = retiredAmount();             else result = normalPayAmount();         };     }     return result; }
  重构之后 double getPayAmount() {     if (_isDead) return deadAmount();     if (_isSeparated) return separatedAmount();     if (_isRetired) return retiredAmount();     return normalPayAmount(); }
  使用场景
  当看到一个方法中,某一层代码块都被一个 if...else 完整控制时,通常可以采用卫语句。 问题三:if…else 表达式过于复杂问题表现
  if...else 所导致的第三个问题来自过于复杂的条件表达式。下面给个简单的例子,当 condition 1、2、3、4 分别为 true、false,请大家排列组合一下下面表达式的结果。
  1 if ((condition1 && condition2 ) || ((condition2 || condition3) && condition4)) { 2    3 }
  我想没人愿意干上面的事情。关键是,这一大坨表达式的含义是什么?关键便在于,当不知道表达式的含义时,没人愿意推断它的结果。
  所以,表达式复杂,并不一定是错。但是表达式难以让人理解就不好了。 如何解决
  对于 if...else 表达式复杂的问题,主要用代码重构中的抽取方法、移动方法等手段解决。因为这些方法在《代码重构》一书中都有介绍,所以这里不再重复。 总结
  本文一个介绍了10种(算上扩展有12种)用于消除、简化 if...else 的方法。还有一些方法,如通过策略模式、状态模式等手段消除 if...else 在《重构与模式》一书中也有介绍。
  正如前言所说,if...else 是代码中的重要组成部分,但是过度、不必要地使用 if...else,会对代码的可读性、可扩展性造成负面影响,进而影响到整个软件系统。
  "干掉"if...else 的能力高低反映的是程序员对软件重构、设计模式、面向对象设计、架构模式、数据结构等多方面技术的综合运用能力,反映的是程序员的内功。要合理使用 if...else,不能没有设计,也不能过度设计。这些对技术的综合、合理地运用都需要程序员在工作中不断的摸索总结。

富士或于今年发布XF18H2S新机也有望年内亮相最近关于富士的新闻主要都是胶片领域相关产品的涨价内容,不过最近在数码无反相机方面也有了新的动向,首先是镜头方面的消息,富士极有可能会在今年年内发布一颗XF18120mmf4变焦镜头除湿机到底应该怎么选?安诗曼260NE除湿机使用效果真实评测每年春天以来,听到淅淅沥沥的雨声,让我们南方的朋友们都感到难受,而这种难受在一年中往往会持续很长时间,尤其是在每年的六月和七月的雨季。空气湿度大,温度高,衣服容易发霉,真的是很糟糕怎样连接共享打印机,永不掉线大家好!我是木田,最近有朋友问我说他家三台电脑,但是只有一台打印机,于是就设置共享使用,但是只要主机那台电脑关机或者重启,其他两台电脑共享就掉线了,需要重新连接,不关机又嫌浪费电,说人话系列英特尔酷睿12代详解(4)摩尔定律之殇火龙888镇楼众所周知,骁龙888系列是个笑话。搭载骁龙888的安卓手机,发热大,耗电快,性能峰值上不去,结合在一起用一个词概括废物。当然这不是高通第一次如此头铁了。上一次这么热,详细比较OPPOReno7和OPPOReno6有什么区别?应该买哪个?OPPOReno7和OPPOReno6是当今流行的OPPO系列中的两款中端智能手机,对于这两款手机许多朋友感到困惑,因为他们不知道这两款手机有什么区别?应该买哪个?接下来,我们将详你怎么选?8000块的iPhone13Pro用4年,还是4000块的安卓用3年?前段时间有媒体报道,现在的年轻人都不换手机了,00后换手机的频率甚至达到了4年。而按照机构的说法,从全球范围来看,大家的换机周期也达到了31个月左右,其中iPhone的平均换机周期有IP68防水才配叫旗舰?实测安卓旗舰防护性能,这些表现简直绝了对于很多人来说,相较于像电池容量拍照效果芯片性能这些能带来明显感知的用机体验而言,手机的防水性能可能并不会显得那么重要,对一些手机所具备的IP68防护级别也不甚了解,使其宛如隐形功Redmi重塑小米三年之前,你要跟我说,红米会变成现在的Redmi,我是不信的。一个曾经追求低价的ODM手机品牌,在2022年都跟奔驰AMG车队联名了,这换谁都不太敢想,但卢伟冰做到了。在我眼里,这复刻机型值得买吗?小米创始人雷军近日在微博上发起了一项你最喜欢哪部小米手机?的调查。结果显示,小米5年前发布的小米6以超过45的压倒性得票率名列榜首,并且在粉丝的评论中又再度出现了不少希望小米复刻这教育部等五部门加强高校在线开放课程教学管理来源人民日报本报北京4月5日电(记者闫伊乔)记者从教育部获悉为规范高校在线开放课程教学管理,近日,教育部中央网信办工业和信息化部公安部市场监管总局等五部门联合印发关于加强普通高等学美联储扣押暗网卖家手中3400万美元比特币美国最大扣押案之一迈阿密是一个以其富裕的生活方式和对加密货币友好的法律和市长而闻名的城市,它已经看到了对使用比特币的网络犯罪分子最积极的打击之一。联邦检察官从佛罗里达州东南部的一名男子手中没收了大约
vivoX70Pro新机正式开售,vivoX60沦为白菜机悲惨让路前几天,vivoX家族的新成员X70系列正式发布,X70X70ProX70Pro三种机型上市。经过几天前的发售,vivoX70ProvivoX70Pro两种机型将于今天发售,分别为亚马逊封店,跨境电商如何应对詹金良文亚马逊的封号潮持续之下,跨境电商这门生意正越来越难做。今年5月以来,美国亚马逊根据卖家行为准则等格式条款,冻结了其平台上的不少中国商家账户,随着亚马逊监管力度的不断升级,相马斯克曾说要做汽车界的丰田,16万的特斯拉Model2要来了早在2018年的特斯拉电池日期间,特斯拉Model2就已经被曝光了。此前有媒体从特斯拉内部人士获得信息称特斯拉上海超级工厂已经开始做好Model2生产准备。而最近有微博网友称特斯拉任正非透露华为新布局,不再从美方获取芯片?自从去年下半年美国规则生效后,芯片短缺就成为了华为的首要难题,直接影响了其各项业务的正常发展,例如智能手机和5G通讯。而就目前来看,华为仍然面临缺芯的压力,美国虽然开放了一些供应商新能源车能否取代燃油车?只是时间的事,比亚迪8月卖了67630辆新能源车能否取代燃油车?这个话题比较有争论性,从目前来看,燃油车不受里程限制,加油站多,适合跑长途。新能源车对于家用代步来说,安静舒适,动力强,节能环保,优点也是挺多的,大部分城市测评完国外加密聊天APP,我决定用回国产软件棱镜门和Facebook用户信息泄露,随着这两个世纪性大事件的爆发,国外网民对信息安全的重视到了史无前例的程度。甚至网友们都在谷歌上疯狂搜索什么类型的加密软件最好用。不仅是国外,国苹果发布会隐藏的黑马ipadmini今年苹果发布会感觉升级最为明显的就是全新升级的ipadmini,升级特别大是发布所有产品里面最具购买影响力的产品没有之一。2021款ipadmini采用的是8。3英寸的LCD材质的iPhone13歧视亚洲人?国内开售之后,依旧供不应求在苹果官方网站关于iPhone13的宣传中,有一张亚洲人的照片,这个照片的特点是眯眯眼单眼皮黝黑皮肤需要注意的是,眯眯眼是对亚洲人的羞辱动作,同时也是对亚洲人的歧视苹果在iPhon奔驰EQE公布,三块大屏无缝对接,还买蔚来吗?在新能源这个大舞台上唱主角的似乎都是一些造车新势力,不过豪华品牌一旦认真起来,杀伤力也是相当强的。在近日的慕尼黑车展上,奔驰展出了旗下多款电动车型,而在这其中最为值得期待的无非是E智能化正在加速杀死混动汽车本文同步发送于同名微信公众号修外求内,欢迎前往关注。前言在人类历史里的工具发展趋势就是降低人类工作负担的过程,汽车作为运输工具的发展史也是降低人类负担的历史,现在汽车要进一步降低人多地启用电动车新国标,电动自行车会成为第二个万亿赛道吗?多地启用电动自行车管理新国标,电动自行车会成第二个万亿赛道吗?未来能与新能源电动汽车齐肩吗?目前看,可能不行。当下电动车如日中天,未来还没来就已经钦定太子,喊着万亿赛道等着继位了那