为什么我们写的代码都是 if-else?

程序员想必都经历过这样的场景:刚开始自己写的代码很简洁,逻辑清晰,函数精简,没有一个 if-else,可随着代码逻辑不断完善和业务的瞬息万变:比如需要对入参进行类型和值进行判断;这里要判断下对象是否为 null;不同类型执行不同的流程。

落地到具体实现只能不停地加 if-else 来处理,渐渐地,代码变得越来越庞大,函数越来越长,文件行数也迅速突破上千行,维护难度也越来越大,到后期基本达到一种难以维护的状态。

虽然我们都很不情愿写出满屏 if-else 的代码,可逻辑上就是需要特殊判断,很绝望,可也没办法避免啊。

其实回头看看自己的代码,写 if-else 不外乎两种场景:异常逻辑处理和不同状态处理。

两者最主要的区别是:异常逻辑处理说明只能一个分支是正常流程,而不同状态处理都所有分支都是正常流程。

怎么理解?举个例子:

//举例一:异常逻辑处理例子
Object obj = getObj();
if (obj != null) {//do something
}else{//do something
}//举例二:状态处理例子
Object obj = getObj();
if (obj.getType == 1) {//do something
}else if (obj.getType == 2) {//do something
}else{//do something
}

第一个例子 if (obj != null) 是异常处理,是代码健壮性判断,只有 if 里面才是正常的处理流程,else 分支是出错处理流程;而第二个例子不管 type 等于 1,2 还是其他情况,都属于业务的正常流程。对于这两种情况重构的方法也不一样。

代码 if-else 代码太多有什么缺点?

缺点相当明显了:最大的问题是代码逻辑复杂,维护性差,极容易引发 bug。如果使用 if-else,说明 if 分支和 else 分支的重视是同等的,但大多数情况并非如此,容易引起误解和理解困难。

是否有好的方法优化?如何重构?

方法肯定是有的。重构 if-else 时,心中无时无刻把握一个原则:

尽可能地维持正常流程代码在最外层。

意思是说,可以写 if-else 语句时一定要尽量保持主干代码是正常流程,避免嵌套过深。

实现的手段有:减少嵌套、移除临时变量、条件取反判断、合并条件表达式等。关注公众号Java核心技术可以获取一份阿里最新的 Java 开发手册。

下面举几个实例来讲解这些重构方法:

异常逻辑处理型重构方法实例一

重构前:

double disablityAmount(){if(_seniority < 2) return 0;if(_monthsDisabled > 12)return 0;if(_isPartTime)return 0;//do somethig}

重构后:

double disablityAmount(){if(_seniority < 2 || _monthsDisabled > 12 || _isPartTime) return 0;//do somethig
}

这里的重构手法叫合并条件表达式:如果有一系列条件测试都得到相同结果,将这些结果测试合并为一个条件表达式。推荐看下:狗屎一样的代码重构。

这个重构手法简单易懂,带来的效果也非常明显,能有效地较少if语句,减少代码量逻辑上也更加易懂。

异常逻辑处理型重构方法实例二

重构前:

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 和最外层并没有关联性的,完全可以提取最顶层。

改为平行关系,而非包含关系,if-else 数量没有变化,但是逻辑清晰明了,一目了然。

另一个重构点是废除了 result 临时变量,直接 return 返回。好处也显而易见直接结束流程,缩短异常分支流程。原来的做法先赋值给 result 最后统一 return,那么对于最后 return 的值到底是那个函数返回的结果不明确,增加了一层理解难度。

总结重构的要点:如果 if-else 嵌套没有关联性,直接提取到第一层,一定要避免逻辑嵌套太深。尽量减少临时变量改用 return 直接返回。

异常逻辑处理型重构方法实例三

重构前:

public double getAdjustedCapital(){double result = 0.0;if(_capital > 0.0 ){if(_intRate > 0 && _duration >0){resutl = (_income / _duration) *ADJ_FACTOR;}}return result;
}

第一步,运用第一招,减少嵌套和移除临时变量:

public double getAdjustedCapital(){if(_capital <= 0.0 ){return 0.0;}if(_intRate > 0 && _duration >0){return (_income / _duration) *ADJ_FACTOR;}return 0.0;
}

这样重构后,还不够,因为主要的语句 (_income / _duration) *ADJ_FACTOR; 在 if 内部,并非在最外层,根据优化原则(尽可能地维持正常流程代码在最外层),可以再继续重构:

public double getAdjustedCapital(){if(_capital <= 0.0 ){return 0.0;}if(_intRate <= 0 || _duration <= 0){return 0.0;}return (_income / _duration) *ADJ_FACTOR;
}

这才是好的代码风格,逻辑清晰,一目了然,没有 if-else 嵌套难以理解的流程。

这里用到的重构方法是:将条件反转使异常情况先退出,让正常流程维持在主干流程。Spring Boot 如何干掉 if else?推荐看下。

异常逻辑处理型重构方法实例四

重构前:

/* 查找年龄大于18岁且为男性的学生列表 */public ArrayList<Student> getStudents(int uid){ArrayList<Student> result = new ArrayList<Student>();Student stu = getStudentByUid(uid);if (stu != null) {Teacher teacher = stu.getTeacher();if(teacher != null){ArrayList<Student> students = teacher.getStudents();if(students != null){for(Student student : students){if(student.getAge() > = 18 && student.getGender() == MALE){result.add(student);}}}else {logger.error("获取学生列表失败");}}else {logger.error("获取老师信息失败");}} else {logger.error("获取学生信息失败");}return result;}

典型的"箭头型"代码,最大的问题是嵌套过深,解决方法是异常条件先退出,保持主干流程是核心流程:

重构后:

/* 查找年龄大于18岁且为男性的学生列表 */public ArrayList<Student> getStudents(int uid){ArrayList<Student> result = new ArrayList<Student>();Student stu = getStudentByUid(uid);if (stu == null) {logger.error("获取学生信息失败");return result;}Teacher teacher = stu.getTeacher();if(teacher == null){logger.error("获取老师信息失败");return result;}ArrayList<Student> students = teacher.getStudents();if(students == null){logger.error("获取学生列表失败");return result;}for(Student student : students){if(student.getAge() > 18 && student.getGender() == MALE){result.add(student);}}return result;}

状态处理型重构方法实例一

重构前:

double getPayAmount(){Object obj = getObj();double money = 0;if (obj.getType == 1) {ObjectA objA = obj.getObjectA();money = objA.getMoney()*obj.getNormalMoneryA();}else if (obj.getType == 2) {ObjectB objB = obj.getObjectB();money = objB.getMoney()*obj.getNormalMoneryB()+1000;}
}

重构后:

double getPayAmount(){Object obj = getObj();if (obj.getType == 1) {return getType1Money(obj);}else if (obj.getType == 2) {return getType2Money(obj);}
}double getType1Money(Object obj){ObjectA objA = obj.getObjectA();return objA.getMoney()*obj.getNormalMoneryA();
}double getType2Money(Object obj){ObjectB objB = obj.getObjectB();return objB.getMoney()*obj.getNormalMoneryB()+1000;
}

这里使用的重构方法是:把 if-else 内的代码都封装成一个公共函数。函数的好处是屏蔽内部实现,缩短 if-else 分支的代码。代码结构和逻辑上清晰,能一下看出来每一个条件内做的功能。

状态处理型重构方法实例二

针对状态处理的代码,一种优雅的做法是用多态取代条件表达式(《重构》推荐做法)。

你手上有个条件表达式,它根据对象类型的不同而选择不同的行为。将这个表达式的每个分支放进一个子类内的覆写函数中,然后将原始函数声明为抽象函数。

重构前:

double getSpeed(){switch(_type){case EUROPEAN:return getBaseSpeed();case AFRICAN:return getBaseSpeed()-getLoadFactor()*_numberOfCoconuts;case NORWEGIAN_BLUE:return (_isNailed)?0:getBaseSpeed(_voltage);}
}

重构后:

class Bird{abstract double getSpeed();
}class European extends Bird{double getSpeed(){return getBaseSpeed();}
}class African extends Bird{double getSpeed(){return getBaseSpeed()-getLoadFactor()*_numberOfCoconuts;}
}class NorwegianBlue extends Bird{double getSpeed(){return (_isNailed)?0:getBaseSpeed(_voltage);}
}

可以看到,使用多态后直接没有了 if-else,但使用多态对原来代码修改过大,需要一番功夫才行。最好在设计之初就使用多态方式。关注公众号Java技术栈可以获取优秀程序员写代码的系列 Java 规范。

总结

if-else 代码是每一个程序员最容易写出的代码,同时也是最容易被写烂的代码,稍不注意,就产生一堆难以维护和逻辑混乱的代码。

针对条件型代码重构把握一个原则:

尽可能地维持正常流程代码在最外层,保持主干流程是正常核心流程。

为维持这个原则:合并条件表达式可以有效地减少if语句数目;减少嵌套能减少深层次逻辑;异常条件先退出自然而然主干流程就是正常流程。

针对状态处理型重构方法有两种:一种是把不同状态的操作封装成函数,简短 if-else 内代码行数;另一种是利用面向对象多态特性直接干掉了条件判断。

现在回头看看自己的代码,犯了哪些典型错误,赶紧运用这些重构方法重构代码吧!!

最后

感谢大家看到最后,如文章有不足,欢迎大家在评论区支持,给予意见。如果觉得我的文章对你有帮助,那就给我一个赞同吧。

每天都会分享java相关技术文章或行业资讯。欢迎大家关注和转发文章。

for里面嵌套if_求求你们了,别再写满屏的 if/ else 了!相关推荐

  1. 求求你们了,别再写满屏的 try catch 了!!

    点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 作者:巨人大哥 来源:cnblogs.com/jurendage/ ...

  2. 没有匹配 if 的非法 else_求求你,别再写这么多if...else...了

    前言 if...else 是所有高级编程语言都有的必备功能.但现实中的代码往往存在着过多的 if...else.虽然 if...else 是必须的,但滥用 if...else 会对代码的可读性.可维护 ...

  3. mave工程中的一个类调用另一个聚合工程的一个类_求求你,别再写上千行的类了好吗...

    专注于Java领域优质技术,欢迎关注 作者:橙味菌 最近在对已有项目进行扩展的时候,发现要改动的一个类它长766行,开放了近40个public接口,我流着泪把它给改完了.为了防止这样的惨剧再次发生在我 ...

  4. 求求你,别再用wait和notify了!

    作者 | 王磊 来源 | Java中文社群(ID:javacn666) 转载请联系授权(微信ID:GG_Stone) Condition 是 JDK 1.5 中提供的用来替代 wait 和 notif ...

  5. 程序员版“我不是药神”:求求领导,别再追查假简历了

    <我不是药神>火的是不得了,相信很多人都看过了,里面有很多经典的台词,也就是这些经典台词,遂衍生出很多"药神体"来吐槽和诉苦,就是这种"某某,我就是想求求你, ...

  6. java switch嵌套if_(新手)Java课程作业,请各位老哥指教:综合运用嵌套if选择结构、switch选择结构、多重if选择结构实现商品换购功能...

    综合运用嵌套if选择结构.switch选择结构.多重if选择结构实现商品换购功能 下面是我自己的代码,功能虽然基本满足,但是感觉好臃肿,很不简洁,有更好的方法吗?import java.util.Sc ...

  7. for循环里面嵌套if_信不信两层python嵌套for循环就能把你搞懵了

    1. 相加为10的组合 任何一门编程语言的基础部分都很容易学习,初学者似乎阅读一遍教程就能够掌握,但这种掌握仅仅停留在理解上,而非运用上,下面是一个简单的两层嵌套for循环,如果你对for循环的理解和 ...

  8. 求求你!别再考秒杀系统了!看完这篇怼回去 ~

    程序员的成长之路 互联网/程序员/技术/资料共享 关注 阅读本文大概需要 25 分钟. 来自:my.oschina.net/xianggao/blog/524943 我们都知道,某厂是最擅长搞秒杀系统 ...

  9. 求求你别再写上千行的类了,试试这些牛逼的重构技巧吧

    欢迎关注方志朋的博客,回复"666"获面试宝典 来源:juejin.cn/post/6844904038383747086 答应我,别再写上千行的类了好吗 最近在对已有项目进行扩展 ...

最新文章

  1. 5300亿参数,SOTA屠榜!最大NLP预训练模型新王登基,微软英伟达联手称霸
  2. hadoop2.4.1+hbase0.98.3实现的分布式网盘系统初步(已开源)
  3. 1.3 程序示例--梯度下降-机器学习笔记-斯坦福吴恩达教授
  4. C++ :sqlite3使用:
  5. Java 10迁移建议
  6. 仅用1天,为湖北黄冈中学搭建直播课堂!
  7. 学平面设计,你必须知道这些提供素材和灵感的地方!
  8. 亚信安全“双引擎”AI技术 亮相2017华为全联接大会
  9. [bzoj1834][ZJOI2010]network 网络扩容
  10. php调用文章至首页,WP如何在首页调用分类文章列表的详细教程
  11. 软件工程设计概念与体系结构设计
  12. html5 游戏营销,五大H5游戏营销成功案例,你都玩过了吗?
  13. Python+旧衣物捐赠系统 毕业设计-附源码290942
  14. 什么是WHQL认证?
  15. 新冠免疫细胞培养、转染、核酸分析整合解决方案
  16. 计算list的字节数/mb数
  17. VK11\VK12\VK13 价格间隔拆分问题
  18. 再谈UI设计的入门与进阶
  19. 量子通信与计算机网络,中科大90后团队搭建 “量子鹊桥”,可将量子通信速率提升四倍,相关研究登上 Nature 封面 | 专访...
  20. 电磁波传播相位是否会变化,关于电磁波的相位不变性和多普勒效应的讨论

热门文章

  1. java 实例域_Java实例域初始化
  2. sparkstreaming自定义kafka
  3. MySql的语法规范
  4. php 开启memcache,php开启与安装 memcache
  5. 小型英语字典(字典训练)
  6. Numpy——numpy的索引
  7. OpenCV3.4.3+Qt5.9.4(QtCreator)开发环境搭建
  8. 剑指offer第二版(150M超清分享PDF+源码)(转)
  9. python aop编程_学习笔记: AOP面向切面编程和C#多种实现
  10. 在Pandas中直接加载MongoDB的数据