位置: 编程技术 - 正文

给PHP开发者的编程指南 第一部分降低复杂程度(给php开发者的编程代码)

发布时间:2024-01-07

推荐整理分享给PHP开发者的编程指南 第一部分降低复杂程度(给php开发者的编程代码),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:php开发文档怎么写,给php开发者的编码是什么,php开发ide,php开发文档怎么写,给php开发者的编程语言,给php开发者的编程语言,给php开发者的编码是什么,给php开发者的编程语言,内容如对您有帮助,希望把文章链接给更多的朋友!

PHP 是一门自由度很高的编程语言。它是动态语言,对程序员有很大的宽容度。作为 PHP 程序员,要想让你的代码更有效,需要了解不少的规范。很多年来,我读过很多编程方面的书籍,与很多资深程序员也讨论过代码风格的问题。具体哪条规则来自哪本书或者哪个人,我肯定不会都记得,但是本文(以及接下来的另一篇文章) 表达了我对于如何写出更好的代码的观点:能经得起考验的代码,通常是非常易读和易懂的。这样的代码,别人可以更轻松的查找问题,也可以更简单的复用代码。降低函数体的复杂度

在方法或者函数体里,尽可能的降低复杂性。相对低一些的复杂性,可以便于别人阅读代码。另外,这样做也可以减少代码出问题的可能性,更易修改,有问题也更易修复。在函数里减少括号数量

尽可能少的使用 if, elseif, else 和 switch 这些语句。它们会增加更多的括号。这会让代码更难懂、更难测试一些(因为每个括号都需要有测试用例覆盖到)。总是有办法来避免这个问题的。代理决策 ("命令,不用去查询(Tell, don't ask)")有的时候 if 语句可以移到另一个对象里,这样会更清晰些。例如:

可以改成: $a->doSomething();这里,具体的判断由 $a 对象的 doSomething() 方法去做了。我们不需要再为此做更多的考虑,只需要安全的调用 doSomething() 即可。这种方式优雅的遵循了命令,不要去查询原则。我建议你深入了解一下这个原则,当你向一个对象查询信息并且根据这些信息做判断的时候都可以适用这条原则。使用map

有时可以用 map 语句减少 if, elseif 或 else 的使用,例如:

可以精简为:

这样使用 map 的方式也让你的代码遵循扩展开放,关闭修改的原则。强制类型

很多 if 语句可以通过更严格的使用类型来避免,例如:

可以通过强制 $a 使用 A 类型来简化:

当然,我们可以通过其他方式来支持 "null" 的情况。这个在后面的文章会提到。Return early

很多时候,函数里的一个分支并非真正的分支,而是前置或者后置的一些条件,就像这样:// 前置条件

这里 if 语句并不是函数执行的一个分支,它只是对一个前置条件的检查。有时我们可以让 PHP 自身来完成前置条件的检查(例如使用恰当的类型提示)。不过,PHP 也没法完成所有前置条件的检查,所以还是需要在代码里保留一些。为了降低复杂度,我们需要在提前知道代码会出错时、输入错误时、已经知道结果时尽早返回。尽早返回的效果就是后面的代码没必要像之前那样缩进了:

像上面这个模板这样,代码会变动更易读和易懂。创建小的逻辑单元

如果函数体过长,就很难理解这个函数到底在干什么。跟踪变量的使用、变量类型、变量声明周期、调用的辅助函数等等,这些都会消耗很多脑细胞。如果函数比较小,对于理解函数功能很有帮助(例如,函数只是接受一些输入,做一些处理,再返回结果)。使用辅助函数在使用之前的原则减少括号之后,你还可以通过把函数拆分成更小的逻辑单元做到让函数更清晰。你可以把实现一个子任务的代码行看做一组代码,这些代码组直接用空行来分隔。然后考虑如何把它们拆分成辅助方法(即重构中的提炼方法)。辅助方法一般是 private 的方法,只会被所属的特定类的对象调用。通常它们不需要访问实例的变量,这种情况需要定义为 static 的方法。在我的经验中,private (static)的辅助方法通常会汇总到分离的类中,并且定义成 public (static 或 instance)的方法,至少在测试驱动开发的时候使用一个协作类就是这种情形。减少临时变量长的函数通常需要一些变量来保存中间结果。这些临时变量跟踪起来比较麻烦:你需要记住它们是否已经初始化了,是否还有用,现在的值又是多少等等。上节提到的辅助函数有助于减少临时变量:

使用辅助方法,我们可以不用临时变量了:

正如你所见,我们把函数变成新函数的组合,这样变得更易懂,也更容易修改。某种方式上,代码还有点符合“扩展开放/修改关闭”,因为我们基本上不需要再修改辅助函数。由于很多算法需要遍历容器,从而得到新的容器或者计算出一个结果,此时把容器本身当做一个“一等公民”并且附加上相关的行为,这样做是很有意义的:

给PHP开发者的编程指南 第一部分降低复杂程度(给php开发者的编程代码)

这样做可以简化函数的组合。虽然减少临时变量通常会带来好的设计,不过上面的例子中也没必要干掉所有的临时变量。有时候临时变量的用处是很清晰的,作用也是一目了然的,就没必要精简。

使用简单的类型

追踪变量的当前取值总是很麻烦的,当不清楚变量的类型时尤其如此。而如果一个变量的类型不是固定的,那简直就是噩梦。数组只包含同一种类型的值 使用数组作为可遍历的容器时,不管什么情况都要确保只使用同一种类型的值。这可以降低遍历数组读取数据的循环的复杂度:

你的代码编辑器也会为你提供数组值的类型提示:

而如果你不能确定 $value 是 DateTime 类型的话,你就不得不在函数里添加前置判断来检查其类型。beberlei/assert库可以让这个事情简单一些:

如果容器里有内容不是 DateTime 类型,这会抛出一个 InvalidArgumentException 异常。除了强制输入相同类型的值之外,使用断言(assert)也是降低代码复杂度的一种手段,因为你可以不在函数的头部去做类型的检查。简单的返回值类型只要函数的返回值可能有不同的类型,就会极大的增加调用端代码的复杂度:

PHP 并不能阻止你返回不同类型的值(或者使用不同类型的参数)。但是这样做只会造成大量的混乱,你的程序里也会到处都充斥着 if 语句。下面是一个经常遇到的返回混合类型的例子:

这个函数会返回 User 对象或者 null,这种做法是有问题的,如果不检查返回值是否合法的 User 对象,我们是不能去调用返回值的方法的。在 PHP 7之前,这样做会造成"Fatal error",然后程序崩溃。下一篇文章我们会考虑 null,告诉你如何去处理它们。可读的表达式

我们已经讨论过不少降低函数的整体复杂度的方法。在更细粒度上我们也可以做一些事情来减少代码的复杂度。隐藏复杂的逻辑

通常可以把复杂的表达式变成辅助函数。看看下面的代码:

可以变得更简单一些,像这样:

阅读代码时可以清楚的知道这个判断依赖 $a, $b 和 $c 三个变量,而函数名也可以很好的表达判断条件的内容。使用布尔表达式if 表达式的内容可以转换成布尔表达式。不过 PHP 也没有强制你必须提供 boolean 值:

$a 会自动转换成 boolean 类型。强制类型转换是 bug 的主要来源之一,不过还有一个问题是会对代码的理解带来复杂性,因为这里的类型转换是隐式的。PHP 的隐式转换的替代方案是显式的进行类型转换,例如:

如果你知道比较的是 bool 类型,就可以简化成这样:

使用 ! 操作符则还可以简化:

不要 Yoda 风格的表达式Yoda 风格的表达式就像这样:

这种表达式主要是为了避免下面的错误:

这里 'hello' 会赋值给 $result,然后成为整个表达式的值。'hello' 会自动转换成 bool 类型,这里会转换成 true。于是 if 分支里的代码在这里会总是被执行。使用 Yoda 风格的表达式可以帮你避免这类问题:

我觉得实际情况下不太会有人出现这种错误,除非他还在学习 PHP 的基本语法。而且,Yoda 风格的代码也有不小的代价:可读性。这样的表达式不太易读,也不太容易懂,因为这不符合自然语言的习惯。

标签: 给php开发者的编程代码

本文链接地址:https://www.jiuchutong.com/biancheng/289126.html 转载请保留说明!

上一篇:PHP基于cookie与session统计网站访问量并输出显示的方法(phpsetcookie)

下一篇:PHP mysql事务问题实例分析(mysql事务执行流程)

  • 车辆使用费属于什么税收分类
  • 增值税抵扣了,附加税怎么算
  • 房地产开发公司排名
  • 信息服务费可以计入办公费吗
  • 暂估入库可以跨年吗
  • 转让土地使用权属于销售无形资产吗
  • 增值税专票逾期抵扣怎么操作
  • 会计凭证传递的终点是
  • 监控系统计入开户费用吗
  • 单位起诉员工赔偿算劳动争议
  • 企业之间现金换承兑合法吗
  • 外币报表折算差额在会计报表中应作为
  • 商业保险可以税前全额扣除吗?
  • 增值税申报金额含税吗
  • 结转增值税的计算公式
  • 专票地址错了可以报销吗
  • 自来水费缴纳后多久来水
  • 加盟费返款怎么入账
  • 个体工商户开票免税额度是多少
  • 保证金计入哪个会计科目
  • 多缴增值税怎么退税
  • 公司需要给员工提供的帮助
  • 公司钱被取走怎么处理
  • 工会经费的会计分录2022
  • 预付款后开发票时的摘要和分录怎么做?
  • 长期待摊费用摊销会计分录
  • mac cpu
  • 外贸企业出口退税计算
  • 借条和欠条的区别 法律效力
  • 小规模纳税人差额征税
  • 融资租赁ppt
  • 纳税评估一般程序包括
  • php返回对象
  • smarty怎么用
  • 应付职工薪酬的含义
  • 购买电脑如何入账
  • js写数组去重
  • 机器学习中的数学——距离定义(八):余弦距离(Cosine Distance)
  • PHP mysqli_free_result()与mysqli_fetch_array()函数详解
  • python输入三个数输出最小的
  • 怎么调整利润分配
  • 融资租赁方式的条件
  • 分公司非独立核算企业所得税处理
  • python next iter
  • 在发票上盖了公章有用吗
  • 商业印花税的计算公式
  • 资产减值准备的举例
  • 无形资产如何计提减值
  • 进口关税免征
  • 收款人是否应当承担还款责任
  • 购买金税盘未抵税怎么办
  • 库存商品视频讲解
  • 企业清算的顺序
  • 商业企业可将商品分为哪三类
  • sql注入神器
  • sqlserver 查看表
  • mysql向指定字段中添加数据
  • Windows Server 2008之数据安全保护
  • 苹果macbook如何录屏
  • wbs是什么的缩写
  • boot process
  • win10系统电脑无法开机怎么办
  • nginx sbin目录
  • win7断电后无法正常启动
  • Win10控制面板打不开
  • linux查看使用率命令
  • linux中使用find命令查找文件
  • javascript中的数字型可以用来保存整数或浮点数(小数)
  • dos常用命令与批处理文件
  • node性能优化
  • html概念及作用
  • jquery设置鼠标样式
  • 同一个页面
  • 如何控制孩子的手机使用时间
  • unity3d怎么用
  • 小规模纳税人收到增值税专用发票怎么办
  • 税控发票开票软件金税盘版口令怎么解锁?
  • 雄安属于北京管吗
  • 如何用微信进行社保认证
  • 财务制度备案表单怎么填
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号