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 相关文章推荐
Content-type 的说明
Oct 09 PHP
php 仿Comsenz安装效果代码打包提供下载
May 09 PHP
JS 网站性能优化笔记
May 24 PHP
php curl的深入解析
Jun 02 PHP
php生成不重复随机数、数组的4种方法分享
Mar 30 PHP
PHP SOCKET编程详解
May 22 PHP
PHPWind9.0手动屏蔽验证码解决后台关闭验证码但是依然显示的问题
Aug 12 PHP
PHP7下协程的实现方法详解
Dec 17 PHP
PHP错误提示It is not safe to rely on the system……的解决方法
Mar 25 PHP
laravel5.6中的外键约束示例
Oct 23 PHP
laravel框架中控制器的创建和使用方法分析
Nov 23 PHP
PHP CURL实现模拟登陆并上传文件操作示例
Jan 02 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
PHP 全角转半角实现代码
2010/05/16 PHP
PHP中生成UUID自定义函数分享
2015/06/10 PHP
laravel 中如何使用ajax和vue总结
2017/08/16 PHP
jquery中获取元素的几种方式小结
2011/07/05 Javascript
基于jquery的滚动鼠标放大缩小图片效果
2011/10/27 Javascript
仿新浪微博返回顶部的jquery实现代码
2012/10/01 Javascript
jquery DIV撑大让滚动条滚到最底部代码
2013/06/06 Javascript
Enter回车切换输入焦点实现思路与代码兼容各大浏览器
2014/09/01 Javascript
精通JavaScript的this关键字
2020/05/28 Javascript
Bootstrap实现下拉菜单效果
2016/04/29 Javascript
AngularJS动态生成div的ID源码解析
2016/08/29 Javascript
H5图片压缩与上传实例
2017/04/21 Javascript
react-router实现跳转传值的方法示例
2017/05/27 Javascript
浅谈react+es6+webpack的基础配置
2017/08/09 Javascript
js单页hash路由原理与应用实战详解
2017/08/14 Javascript
微信小程序下拉刷新界面的实现
2017/09/28 Javascript
深入剖析JavaScript instanceof 运算符
2019/06/14 Javascript
解决vue做详情页跳转的时候使用created方法 数据不会更新问题
2020/07/24 Javascript
vue data对象重新赋值无效(未更改)的解决方式
2020/07/24 Javascript
解决vue单页面多个组件嵌套监听浏览器窗口变化问题
2020/07/30 Javascript
Postman参数化实现过程及原理解析
2020/08/13 Javascript
让Vue响应Map或Set的变化操作
2020/11/11 Javascript
[02:19]DOTA选手解说齐贺岁
2018/02/11 DOTA
python实现探测socket和web服务示例
2014/03/28 Python
使用pdb模块调试Python程序实例
2015/06/02 Python
详解Python文件修改的两种方式
2019/08/22 Python
python GUI库图形界面开发之pyinstaller打包python程序为exe安装文件
2020/02/26 Python
IntelliJ 中配置 Anaconda的过程图解
2020/06/01 Python
澳大利亚珠宝商:Shiels
2019/10/06 全球购物
英国医生在线预约:Top Doctors
2019/10/30 全球购物
关于运动会的稿件
2014/02/02 职场文书
美丽家庭事迹材料
2014/05/03 职场文书
工作失职检讨书500字
2014/10/17 职场文书
给女朋友道歉的话大全
2015/01/20 职场文书
幼儿园五一劳动节活动总结
2015/02/09 职场文书
学习十八大的感悟
2015/08/11 职场文书