`

你还在用if else吗?

阅读更多

软件编程牛人网

面向过程设计和面向对象设计的主要区别是:是否在业务逻辑层使用冗长的if else卸稀H绻慊乖诖罅渴褂胕f else,当然,界面表现层除外,即使你使用Java/C#这样完全面向对象的语言,也只能说明你的思维停留在传统的面向过程语言上。

传统思维习惯分析

  为什么会业务逻辑层使用if else,其实使用者的目的也是为了重用,但是这是面向过程编程的重用,程序员只看到代码重用,因为他看到if else几种情况下大部分代码都是重复的,只有个别不同,因此使用if else可以避免重复代码,并且认为这是模板Template模式。

  他范的错误是:程序员只从代码运行顺序这个方向来看待它的代码,这种思维类似水管或串行电路,水沿着水管流动(代码运行次序),当遇到几个分管(子管),就分到这几个分管子在流动,这里就相当于碰到代码的if else处了。

  而使用OO,则首先打破这个代码由上向下顺序等同于运行时的先后循序这个规律,代码结构不由执行循序决定,由什么决定呢?由OO设计;设计模式会取代这些if else,但是最后总是由一个Service等总类按照运行顺序组装这些OO模块,只有一处,这处可包含事务,一般就是Service,EJB中是Session bean。

  一旦需求变化,我们更多的可能是Service中各个OO模块,甚至是只改动Service中的OO模块执行顺序就能符合需求。

  这里我们也看到OO分离的思路,将以前过程语言的一个Main函数彻底分解,将运行顺序与代码其他逻辑分离开来,而不是象面向过程那样混乱在一起。所以有人感慨,OO也是要顺序的,这是肯定的,关键是运行顺序要单独分离出来。

  是否有if else可以看出你有没有将运行顺序分离到家。

设计模式的切入口

  经常有人反映,设计模式是不错,但是我很难用到,其实如果你使用if else来写代码时(除显示控制以外),就是在写业务逻辑,只不过使用简单的判断语句来作为现实情况的替代者。

   还是以大家熟悉的论坛帖子为例子,如ForumMessage是一个模型,但是实际中帖子分两种性质:主题贴(第一个根贴)和回帖(回以前帖子的帖子),这里有一个朴素的解决方案:
建立一个ForumMessage,然后在ForumMessage加入isTopic这样判断语句,注意,你这里一个简单属性的判断引入,可能导致你的程序其他地方到处存在if else 的判断。

  如果我们改用另外一种分析实现思路,以对象化概念看待,实际中有主题贴和回帖,就是两种对象,但是这两种对象大部分是一致的,因此,我将ForumMessage设为表达主题贴;然后创建一个继承ForumMessage的子类ForumMessageReply作为回帖,这样,我在程序地方,如Service中,我已经确定这个Model是回帖了,我就直接下溯为ForumMessageReply即可,这个有点类似向Collection放入对象和取出时的强制类型转换。通过这个手段我消灭了以后程序中if else的判断语句出现可能。

  从这里体现了,如果分析方向错误,也会导致误用模式。

  讨论设计模式举例,不能没有业务上下文场景的案例,否则无法决定是否该用模式,下面举两个对比的例子:

  第一. 这个帖子中举例的第一个代码案例是没有上下文的,文中只说明有一段代码:

main() {

if(case A){

//do with strategy A

}else(case B){

//do with strategy B

}else(case C){

//do with strategy C

}

}

 

  这段代码只是纯粹的代码,没有业务功能,所以,在这种情况下,我们就很难确定使用什么模式,就是一定用策略模式等,也逃不过还是使用if else的命运,设计模式不是魔法,不能将一段毫无意义的代码变得简单了,只能将其体现的业务功能更加容易可拓展了。

  第二.在这个帖子中,作者举了一个PacketParser业务案例,这段代码是体现业务功能的,是一个数据包的分析,作者也比较了各种模式使用的不同,所以我们还是使用动态代理模式或Command模式来消灭那些可能存在的if else

  由以上两个案例表明:业务逻辑是我们使用设计模式的切入点,而在分解业务逻辑时,我们习惯则可能使用if else来实现,当你有这种企图或者已经实现代码了,那么就应该考虑是否需要重构Refactoring了。

if else替代者

  那么实战中,哪些设计模式可以替代if else呢?其实GoF设计模式都可以用来替代if else,我们分别描述如下:

  • 状态模式 
      当数据对象存在各种可能性的状态,而且这种状态将会影响到不同业务结果时,那么我们就应该考虑是否使用状态模式,当然,使用状态模式之前,你必须首先有内存状态这个概念,而不是数据库概念,因为在传统的面向过程的/面向数据库的系统中,你很难发现状态的,从数据库中读取某个值,然后根据这个值进行代码运行分流,这是很多初学者常干的事情。参考文章:状态对象:数据库的替代者
      使用传统语言思维的情况还有:使用一个类整数变量标识状态:

    public class Order{

    private int status;

    //说明:

    //status=1 表示订货但为查看 ;

    //status=2 表示已经查看未处理;

    //status=3 表示已经处理未付款

    //status=4 表示已经付款未发货

    //status=5 表示已经发货

    }


      上述类设计,无疑是将类作为传统语言的函数来使用,这样导致程序代码中存在大量的if else。

  • 策略模式 
      当你面临几种算法或者公式选择时,可以考虑策略模式,传统过程语言情况是:从数据库中读取算法数值,数值1表示策略1,例如保存到数据库;数值为2表示策略2,例如保存到XMl文件中。这里使用if else作为策略选择的开关。 

  • command模式 
      传统过程的思维情况是:如果客户端发出代号是1或"A",那么我调用A.java这个对象来处理;如果代号是2或"B",我就调用B.java来处理,通过if else来判断客户端发送过来的代码,然后按事先约定的对应表,调用相应的类来处理。

  • MVC模式 
      MVC模式的传统语言误用和Command模式类似,在一个Action类中,使用if else进行前后台调度,如果客户端传送什么命令;我就调用后台什么结果;如果后台处理什么结构,再决定推什么页面,不过,现在我们使用Struts/JSF这样MVC模式的框架实现者就不必范这种低级错误。

  • 职责链模式 
      职责链模式和Command模式是可选的,如果你实在不知道客户端会发出什么代号;也没有一个事先定义好的对照表,那么你只能编写一个个类去碰运气一样打开这个包看一下就可以。与Command是不同在AOP vs Decorator一文中有分析。

  • 代理或动态代理模式 
      代理对象可以是符合某种条件的代表者,比如,权限检验,传统面向过程思维是:当一个用户登陆后,访问某资源时,使用if else进行判断,只有某种条件符合时,才能允许访问,这样权限判断和业务数据逻辑混乱在一起,使用代理模式可以清晰分离,如果嫌不太好,使用动态代理,或者下面AOP等方式。

  • AOP或Decorator模式
      
      其实使用filter过滤器也可以替代我们业务中的if else,过滤器起到一种过滤和筛选作用,将符合本过滤器条件的对象拦截下来做某件事情,这就是一个过滤器的功能,多个过滤器组合在一起实际就是if else的组合。
      所以,如果你实在想不出什么办法,可以使用过滤器,将过滤器看成防火墙就比较好理解,当客户端有一个请求时,经过不同性质的防火墙,这个防火墙是拦截端口的;那个防火墙是安全检查拦截等等。过滤器也如同红蓝白各种光滤镜;红色滤镜只能将通过光线中的红色拦截了;蓝色滤镜将光线中的蓝色拦截下来,这实际上是对光线使用if else进行分解。


      如图,通过一个个条件过滤器我们立体地实现了对信号的分离,如果你使用if else,说明你是将图中的条件1/2/3/4合并在一起,在同一个地方实现条件判断。
      需要深入了解过滤器的实现细节和微小区别,请参考文章:AOP vs Decorator

    OO设计的总结  

      还有一种伪模式,虽然使用了状态等模式,但是在模式内部实质还是使用if else或switch进行状态切换或重要条件判断,那么无疑说明还需要进一步努力。更重要的是,不能以模式自居,而且出书示人。

      真正掌握面向对象这些思想是一件困难的事情,目前有各种属于揪着自己头发向上拔的解说,都是误人子弟的,所以我觉得初学者读Thinking in Java(Java编程思想)是没有用,它试图从语言层次来讲OO编程思想,非常失败,作为语言参考书可以,但是作为Java体现的OO思想的学习资料,就错了。

      OO编程思想是一种方法论,方法论如果没有应用比较,是无法体会这个方法论的特点的,禅是古代一个方法论,悟禅是靠挑水砍柴这些应用才能体会。

      那么OO思想靠什么应用能够体会到了?是GoF设计模式,GoF设计模式是等于软件人员的挑水砍柴等基本活,所以,如果一个程序员连基本活都不会,他何以自居OO程序员?从事OO专业设计编程这个工作,如果不掌握设计模式基本功,就象一个做和尚的人不愿意挑水砍柴,他何以立足这个行业?早就被师傅赶下山。

      最后总结:将if else用在小地方还可以,如简单的数值判断;但是如果按照你的传统习惯思维,在实现业务功能时也使用if else,那么说明你的思维可能需要重塑,你的编程经验越丰富,传统过程思维模式就容易根深蒂固,想靠自己改变很困难;建议接受专业头脑风暴培训

      用一句话总结:如果你做了不少系统,很久没有使用if else了,那么说明你可能真正进入OO设计的境地了。(这是本人自己发明的实战性的衡量考核标准)

  •  

     

     

    分享到:
    评论

    相关推荐

      使用JS枚举简化if else

      在处理条件判断时,通常我们会使用if...else或switch语句。以下是一个使用if...else的经典例子: ```javascript function printColor(color) { if (color === 'red') { console.log('红色'); } else if (color =...

      if-elseif-else使用范例

      在这个"if-elseif-else使用范例"中,我们可以从给出的五个`.cpp`文件(作业1至作业5)中学习到如何有效地应用这种控制结构。 首先,`if`语句是条件判断的基础,它的基本语法如下: ```cpp if (condition) { // 当...

      代码中大量的 if else如何优化.doc

      在编程中,我们常常会遇到大量的 if else 语句,这些语句会使得代码变得难以维护和阅读。那么,如何优化这些 if else 语句呢?下面我们来讨论一些常见的优化方法。 首先,让我们看一下最简单的优化方法:提前 ...

      delphi if else 语句使用方法

      ### Delphi IF ELSE 语句使用方法详解 在Delphi编程语言中,`IF...ELSE`...以上就是关于Delphi中`IF...ELSE`语句的基本介绍和使用方法,希望对你有所帮助。在实际编程中,合理运用这些结构可以使程序更加灵活高效。

      if-else 三地址

      本篇文章将深入探讨如何使用递归下降法来设计一个翻译程序,将`if-else`条件语句转化为三地址码。 首先,我们需要理解`if-else`语句的基本结构。在大多数高级编程语言中,`if-else`语句用于基于某个条件执行不同的...

      运用IF-else

      标题中的“运用IF-else”指的是在编程语言中使用条件语句进行逻辑判断的基本概念。在编程中,IF-else结构是控制流程的一种关键方式,它允许程序根据特定条件执行不同的代码块。当我们需要在满足某种条件时执行特定...

      if else 语句 sql server

      if else 语句

      VS2015 if else for 代码折叠

      VS2015 if else for 代码折叠,VS2015 if else for 代码折叠,VS2015 if else for 代码折叠,VS2015 if else for 代码折叠,VS2015 if else for 代码折叠

      verilog中多个else_if级联造成的综合电路的低效率及解决办法

      ### Verilog中多个else_if...通过使用多个`if_else`或`case`语句等方法替代`else_if`级联,可以在很大程度上优化这些问题,从而得到更加高效的硬件设计。对于工程师来说,理解这些细节并在实践中灵活运用是非常重要的。

      C语言中if-else语句的嵌套分析.pdf

      if-else语句可以嵌套使用,即一个if语句可以包含另一个if语句。这时,需要注意if语句之间的嵌套关系。嵌套关系可以分为两种情况: (1)并列关系:if语句之间没有包含关系,但if语句的执行顺序会影响变量的当前值。...

      if_else翻译程序

      在编程领域,`if-else`条件语句是控制流程中的关键组成部分,它允许程序员根据特定条件执行不同的代码块。本项目"if_else翻译程序"专注于设计一个能够解析并翻译`if-else`结构的程序,采用LR方法,并通过输出四元式...

      Java基础if和ifelse选择结构

      在Java编程语言中,`if` 和 `if-else` 结构是控制流程的重要组成部分,用于根据特定条件执行不同的代码块。本文将详细探讨这些结构,并通过实例帮助读者更好地理解如何使用它们。 #### 一、运算符分类 在Java中,...

      python if-else 写在一行(csdn)————程序.pdf

      在给定的例子中,我们有两个场景展示了如何在一行内使用`if-else`语句。 场景一:将二维列表中的小于15的元素替换为`None`,大于15的元素忽略。 ```python data = [[1,2,3,18,19], [4,5,6,20,21]] res = [[t if t ...

      语法分析-if-else的匹配问题

      - **使用注释**:在复杂的if-else结构中添加注释,以解释每个部分的目的和作用。 在进行语法分析时,开发人员和工具应该能够检测到这些匹配问题并给出相应的错误提示。修复这些问题可以确保代码的正确执行,并避免...

      C语言中分支结构:if语句、if-else语句和switch语句的使用方法.docx

      C语言中分支结构:if语句、if-else语句和switch语句的使用方法 C语言中分支结构是一种非常基本但非常重要的编程结构,它允许程序根据条件执行不同的代码块。在C语言中,分支结构有三种基本类型:if语句、if-else...

      Java学习,if--else练习

      在"Java学习,if--else练习"这个资源中,我们可以深入理解并掌握如何有效地使用`if`和`else`来编写条件判断。 首先,`if`语句允许我们检查一个布尔表达式,如果表达式的值为`true`,则执行`if`后面的代码块。基本...

      使用if else语句把它们的中间数找出来

      在这个特定的题目中,我们需要编写一个程序,接收三个实数(浮点数)作为输入,然后使用`if...else`语句找到并输出这三个数中的中间值。中间值指的是这三个数中数值大小居中的那个数,即它既不大于另外两个数,也不...

      asp.net if else 语句

      asp.net---- if else 语句

      语句注解IF_ELSE语句

      4. **嵌套**:`if-else`语句可以被嵌套,即在一个`if`或`else`代码块内部还可以包含另一个`if-else`语句。嵌套时,`else`和`else if`遵循就近原则,即与最近的未被匹配的`if`语句匹配。 #### 示例代码 以下是一个...

      Objective-C else if 范例

      在这里,我们深入探讨`else if`在Objective-C中的使用及其相关知识点。 `else if`结构常用于当一个条件不满足时,检查另一个或多个条件的情况。它的基本形式如下: ```objc if (condition1) { // 如果condition1...

    Global site tag (gtag.js) - Google Analytics