PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式


Posted in PHP onApril 09, 2012

思维导图
点击下图,查看大图。

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 介绍
 
 条件逻辑有可能十分复杂,因此本章提供一些重构的手法,专门用来简化它们。
 
全文简述(你可直接跳过下面的内容)

核心重构:Decompose Conditional——分离”转辙逻辑“(switching logic)和”操作细节“(details)分离。

多处测试有相同结果:Consolidate Conditional Expresssion

条件代码中去掉重复成分:Consolidate Duplicate

标识特殊情况:Replace Nested Conditional with Guard Clauses

去除讨厌的控制标记:Remove Control Flag

 
 
 专业术语
 
decompose:分解,分离
consolidate:合并
eligible:合适的,合格的
fragment:碎片,片段
nest:嵌套
guard:保卫
clause:从句
polymorphism:多态
assertion:断言
unchecked exception:不可控异常
 
 Decompose Conditional
 
状况:你有一个复杂的条件(if-else if-else)语句,那么从if、else if、else三个段落中分别提炼出函数。

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 

 

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 

 Consolidate Conditional Expression
 
状况:你有一些条件测试,都得到相同的结果,那么将这些测试合并为一个条件式,并将这个条件提炼称为一个独立的函数。
动机: 1、合并后的条件代码会告诉你“实际上只有一次条件检查,只不过有数个并列条件需要检查而已“,——使检查的用意更清晰。

   2、为Extract Method做好准备。——将检查条件提炼成一个独立函数,对于理清代码意义非常有用。它把描述“做什么”的语句换成了“为什么这样做”。

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 

条件语句的“合并理由”也同时指出了“不要合并”的理由:如果你认为你的这些检查的确彼此独立,的确不应该被视为同一次检查,那么就不要使用本项重构。因为在这种情况下,你的代码已经清楚表达出自己的意义。

 

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 

 Consolidate Duplicate Conditional Fragments
 
状况:在条件式的每个分支上有着相同的一段代码,那么将这段重复代码搬移到条件之外。
 

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 
 Remove Control Flag
 
状况:在一系列布尔表达式中,某个变量带有“控制标记”的作用,那么以break语句或return语句取代控制标记。
 

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式
 

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 
 Replace Nested Conditional with Guard Clauses
 
状况:函数中的条件逻辑使人很难看清正常的执行路径,那么使用卫语句(Guard Clauses)表现所有特殊情况。

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式


条件式的两种形式:

1、所有分支都属于正常行为:使用[if ... else..]

2、条件式极其罕见:应该单独检查该条件,并在该条件为真时,立刻从函数中返回。——这样的单独检查常常被称为”卫语句“

Replace Nested Conditional with Guard Clauses精髓:给某一分支以特别重视。

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 
 Replace Conditional with Polymorphism
 
状况:你手上有个表达式,它根据对象型别的不同而选择不同的行为,那么将这个条件式的每个分支放进一个subclass内的覆写函数中,然后将原始函数声明为抽象函数。
 

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

此代码的坏味道:

1、它太长,当视频有新类型的时候,它会变得更长。

2、它明显做了不止一件事。

3、它违反了单一权责原则,因为它有好几个修改它的理由。

4、它违反了开放闭合原则,因为每当添加新类型时,必须修改它。不过最麻烦的可能是到处皆有类似结构(_get类型名Rank())的函数。

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 Introduce Assertion
 
状况:某一段代码需要对程序状态(state)做出某种假设,那么以断言(assertion)明确表现这种假设。
 

 

 

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

运行结果:

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

运行结果:

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

 
采点:

PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式

1、常常会有这样的代码,只有当某个条件为真时,该段代码才能正常运行。——实际上程序最后成品往往将assertion统统删除。

2、这样的假设通常并没有在代码中明确表现出来,你必须阅读整个算法才能看出。——有时候程序员会以注释写出这样的假设,而assetion是一种更好的技术。

3、assertion是一个条件式,应该总是为真。如果失败,表示程序员犯了错误

4、assertion可以作为交流与调试的辅助。——交流:可以帮助程序员阅读理解代码所做的假设。调试:帮助程序员找到bug,可以在距离最近的地方抓住bug。

5、assertion并不改变程序的任何行为。

6、assertion价值:帮助程序员理解代码正确运行的必要条件。

7、建议最好把assertion的条件式使用Extract Method,为了将若干地方的重复码提炼到同一个函数中,也许只是为了更清楚说明条件式的用途。

 
 总结
 
       这一章我比较喜欢“Replace Nested Conditional with Guard Clauses “这个方式,我在平时的代码中也经常这样用,还有人给这种方式取名叫”卫从句“。
      还有一个就是我经常在php开发中用的调试是var_dump()或print_r(),我也第一次发现php中还有assert这种方式,不错!
 
        在学习和实践的过程中,我也学到了很多不错的方式。但是我觉得在团队开发中,有的时候还是”大局为重“,按照团队的习惯方式去编码,或者你可以跟团队沟通,得到大家的认可之后,在使用这里面的方法,这样大家彼此调试和阅读对方代码的时候比较方便。
PHP 相关文章推荐
一个多文件上传的例子(原创)
Oct 09 PHP
php设计模式 Interpreter(解释器模式)
Jun 26 PHP
Php图像处理类代码分享
Jan 19 PHP
解析PHP无限级分类方法及代码
Jun 21 PHP
php使用filter过滤器验证邮箱 ipv6地址 url验证
Dec 25 PHP
PHP移动文件指针ftell()、fseek()、rewind()函数总结
Nov 18 PHP
php实现的通用图片处理类
Mar 24 PHP
PHP中trim()函数简单使用指南
Apr 16 PHP
浅谈PHP定义命令空间的几个注意点(推荐)
Oct 29 PHP
PHP MYSQL简易交互式站点开发
Dec 27 PHP
利用PHP扩展Xhprof分析项目性能实践教程
Sep 05 PHP
PHP实现的多进程控制demo示例
Jul 22 PHP
PHP 杂谈《重构-改善既有代码的设计》之三 重新组织数据
Apr 09 #PHP
PHP 杂谈《重构-改善既有代码的设计》之一 重新组织你的函数
Apr 09 #PHP
PHP乱码问题,UTF-8乱码常见问题小结
Apr 09 #PHP
PHP中return 和 exit 、break和contiue 区别与用法
Apr 09 #PHP
php 的加密函数 md5,crypt,base64_encode 等使用介绍
Apr 09 #PHP
PHP压缩html网页代码(清除空格,换行符,制表符,注释标记)
Apr 02 #PHP
PHP文件注释标记及规范小结
Apr 01 #PHP
You might like
让你的网站首页自动选择语言转跳
2006/12/06 PHP
php设计模式之简单工厂模式详解
2014/09/04 PHP
php短网址和数字之间相互转换的方法
2015/03/13 PHP
详解PHP的Yii框架中自带的前端资源包的使用
2016/03/31 PHP
thinkphp配置文件路径的实现方法
2016/08/30 PHP
php简单压缩css样式示例
2016/09/22 PHP
javascript 流畅动画实现原理
2009/09/08 Javascript
JS实现商品倒计时实现代码
2013/05/03 Javascript
jquery ajax,ashx,json的用法总结
2014/02/12 Javascript
IE浏览器中图片onload事件无效的解决方法
2014/04/29 Javascript
js判断当页面无法回退时关闭网页否则就history.go(-1)
2014/08/07 Javascript
javascript学习笔记(六)数据类型和JSON格式
2014/10/08 Javascript
javascript带回调函数的异步脚本载入方法实例分析
2015/07/02 Javascript
Vue.js 单页面多路由区域操作的实例详解
2017/07/17 Javascript
Vee-Validate的使用方法详解
2017/09/22 Javascript
vue实现与安卓、IOS交互的方法
2018/11/02 Javascript
vue自定义指令之面板拖拽的实现
2019/04/14 Javascript
vue使用高德地图根据坐标定位点的实现代码
2019/08/22 Javascript
vue实现全屏滚动效果(非fullpage.js)
2020/03/07 Javascript
使用Python编写Linux系统守护进程实例
2015/02/03 Python
Python实现国外赌场热门游戏Craps(双骰子)
2015/03/31 Python
python实现搜索本地文件信息写入文件的方法
2016/02/22 Python
Python爬取数据并写入MySQL数据库的实例
2018/06/21 Python
FFT快速傅里叶变换的python实现过程解析
2019/10/21 Python
Django日志及中间件模块应用案例
2020/09/10 Python
python 进程池pool使用详解
2020/10/15 Python
详解使用postMessage解决iframe跨域通信问题
2019/11/01 HTML / CSS
工程总经理工作职责
2013/12/09 职场文书
光盘行动倡议书
2014/02/02 职场文书
雷人标语集锦
2014/06/19 职场文书
大学生实习证明范本
2014/09/19 职场文书
合理化建议书范文
2015/09/14 职场文书
golang判断key是否在map中的代码
2021/04/24 Golang
一篇文章带你搞懂Python类的相关知识
2021/05/20 Python
eclipse创建项目没有dynamic web的解决方法
2021/06/24 Java/Android
mysql性能优化以及配置连接参数设置
2022/05/06 MySQL