PHP 编程的 5个良好习惯


Posted in PHP onFebruary 20, 2009

根据具体的情况,一般的开发人员往往比优秀的开发人员的效率低 10%~20%。优秀的开发人员的效率更高,因为他们拥有丰富的经验和良好的编程习惯。不良的编程习惯将会影响到效率。本文通过展示一些良好的编程习惯,帮助您成为更优秀的程序员。

   这些良好的编程习惯不仅能提高效率,还能让您编写出在应用程序的整个生命周期中易于维护的代码。编写出来的代码可能需要大量的维护;应用程序的维护是一笔很大的开支。养成良好的编程习惯能够提高设计质量(比如模块化),从而使代码更加容易理解,因此维护就更加容易,同时也降低维护成本。

不良的编程习惯会造成代码缺陷,使其难以维护和修改,并且很可能在修改时又引入其他缺陷。以下是 5 个良好的编程习惯,能够帮助 PHP 代码避免这些缺陷:

  1. 使用良好的命名。
  2. 分成更小的部分。
  3. 为代码添加注释。
  4. 处理错误条件。
  5. 切忌使用复制粘贴。

下一小节将详细介绍这些习惯。

使用良好的命名

  使用良好的命名是最重要的编程习惯,因为描述性强的名称让代码更加容易阅读和理解。代码是否好理解取决于是否能在未来维护它。即便代码不带有注释,如果它很容易理解,将大大方便日后的更改。这个习惯的目标是让您编写的代码像书本一样容易阅读和理解。

不良习惯:含糊的或无意义的名称

  清单 1 中的代码包含过短的变量名、难以辨认的缩写词,并且方法名不能反映该方法的功能。如果方法名给人的感觉是它应该做这件事情,而实际中它却做另外的事情,这将带来严重的问题,因为它会误导人。

清单 1. 不良习惯:含糊的或无意义的名称

良好习惯:说明性强并且简洁的名称

  清单 2 中的代码体现了良好的编程习惯。新的方法名具有很强的说明性,反映了方法的用途。同样,更改后的变量名也更具说明性。惟一的保持最短的变量是$i,在本清单中,它是一个循环变量。尽管很多人不赞同使用过短的名称,但在循环变量中使用还是可以接受的(甚至有好处),因为它明确表明了代码的功能。

清单 2. 良好习惯:说明性强并且简洁的名称

  我们鼓励您将大的条件拆分为一个方法,然后用能够描述该条件的名字命名方法。这个技巧能够提高代码的可读性,并且能够将条件具体化,使之能够被提取甚至重用。如果条件发生变化,更新方法也很容易。因为方法拥有一个有意义的名字,所以它能反映代码的用途,让代码更容易阅读。

 

分成更小的部分

  专心解决一个问题之后再继续编程,这样会让您更轻松。在解决一个紧急的问题时,如果继续编程,会使函数越来越长。从长远来说,这并不是一个问题,但您要记得回过头来将它重构为更小的部分。

  重构是个不错的主意,但您应该养成编写更短、功能更集中的代码。短的方法能够在一个窗口中一次看完,并且容易理解。如果方法过长,不能在一个窗口中一次看完,那么它就变得不容易理解,因为您不能快速地从头到尾了解它的整个思路。

  构建方法时,您应该养成这样的习惯,让每个方法只完成一件事情。这个习惯很好,因为:首先,如果方法只完成一件事情,那么它就更容易被重用;其次,这样的方法容易测试;第三,这样的方法便于理解和更改。

不良习惯:过长的方法(完成很多件事情)

  清单 3 展示了一个很长的函数,其中存在很多问题。它完成很多件事情,因此不够紧凑。它也不便于阅读、调试和测试。它要做的事情包括遍历一个文件、构建一个列表、为每个对象赋值、执行计算等等。

清单 3. 不良习惯:过长的函数

" ."" ."My Great Feed" ."http://www.example.com/feed.xml" ."The best feed in the world" ."en-us" ."Tue, 20 Oct 2008 10:00:00 GMT" ."Tue, 20 Oct 2008 10:00:00 GMT" ."http://www.example.com/rss" ."MyFeed Generator" ."editor@example.com" ."webmaster@example.com" ."5";// build the feed...while ($row = mysql_fetch_assoc($result)) {$title = $row['title'];$link = $row['link'];$description = $row['description'];$date = $row['date'];$guid = $row['guid'];$feed .= "";$feed .= "" . $title . "";$feed .= "" . $link . "";$feed .= " " . $description . "";$feed .= "" . $date . "";$feed .= "" . $guid . "";$feed .= "";}$feed .= "";// write the feed out to the server...echo($feed);}?>

  如果多编写几个这样的方法,维护就成了真正的难题了。

良好习惯:易管理、功能专一的方法

  清单 4 将原来的方法改写为更加紧凑、易读的方法。在这个示例中,将一个很长的方法分解为几个短方法,并且让每个短方法负责一件事情。这样的代码对将来的重用和测试都是大有裨益的。

清单 4. 良好习惯:易管理、功能专一的方法

" ."" ."My Great Feed" ."http://www.example.com/feed.xml" ."The best feed in the world" ."en-us" ."Tue, 20 Oct 2008 10:00:00 GMT" ."Tue, 20 Oct 2008 10:00:00 GMT" ."http://www.example.com/rss" ."MyFeed Generator" ."editor@example.com" ."webmaster@example.com" ."5";}function createRssFooter(){return "";}function createRssItem($title, $link, $desc, $date, $guid){$item .= "";$item .= "" . $title . "";$item .= "" . $link . "";$item .= " " . $description . "";$item .= "" . $date . "";$item .= "" . $guid . "";$item .= "";return $item;}function getUserMaxStories($db_link, $default){$perfsQuery = sprintf("SELECT max_stories FROM user_perfs WHERE user= '%s'",mysql_real_escape_string($user));$result = mysql_query($perfsQuery, $db_link);$max_stories = $default;if ($row = mysql_fetch_assoc($result)) {$max_stories = $row['max_stories'];}return $max_stories;}function writeRssFeed($user){// Get the DB connection information$settings = parse_ini_file("rss_server.ini");// look up the user's preferences...$link = mysql_connect($settings['db_host'], $settings['user'],$settings['password']) OR die(mysql_error());$max_stories = getUserMaxStories($link, 25);// go get my data$newsQuery = sprintf("SELECT * FROM stories WHERE post_date = '%s'",mysql_real_escape_string(time()));$result = mysql_query($newsQuery, $link);$feed = createRssHeader();$i = 0;// build the feed...while ($row = mysql_fetch_assoc($result)) {if ($i

 

  将长方法拆分为短方法也是有限制的,过度拆分将适得其反。因此,不要滥用这个良好的习惯。将代码分成大量的片段就像没有拆分长代码一样,都会造成阅读困难。

 

为代码添加注释

  要为代码添加良好的注释有时似乎和编写代码一样难。要了解应该为哪些内容添加注释并不容易,因为我们常常倾向于注释代码当前做的事情。注释代码的目的是不错的主意。在函数的不是很明显的头部代码块中,告诉读者方法的输入和输出,以及方法的最初目标。

  注释代码当前做什么是很常见的,但这是不必要的。如果代码很复杂,不得不注释它当前在做什么,这将暗示您应该重写代码,让它更容易理解。学会使用良好的名称和更短的方法,在不提供注释说明其用途的情况下提高代码的可读性。

不良习惯:函数注释过多或不足

  清单 5 中的注释仅告诉读者代码在做什么 — 它正在通过一个循环进行迭代或添加一个数字。但它忽略了它为什么做当前的工作。这使维护该代码的人员不知道是否可以安全地更改代码(不引入新缺陷)。

清单 5. 不良习惯:函数注释过多或不足

severity = $sev;$this->message = $msg;}public function getSeverity(){return $this->severity;}public function setSeverity($severity){$this->severity = $severity;}public function getMessage(){return $this->message;}public function setMessage($msg){$this->message = $msg;}}function cntMsgs($messages){$n = 0;/* iterate through the messages... */foreach($messages as $m) {if ($m->getSeverity() == 'Error') {$n++; // add one to the result;}}return $n;}$messages = array(new ResultMessage("Error", "This is an error!"),new ResultMessage("Warning", "This is a warning!"),new ResultMessage("Error", "This is another error!"));$errs = cntMsgs($messages);echo("There are " . $errs . " errors in the result.\n");?>

良好习惯:带注释的函数和类

  清单 6 中的注释告诉读者类和方法的目的。该注释解释了为什么代码在做当前的工作,这对未来维护代码十分有用。可能需要根据条件变更而修改代码,如果能够轻松了解代码的目的,则修改起来很容易。

清单 6. 良好习惯:带注释的函数和类

severity = $sev;$this->message = $msg;}/*** Returns the severity of the message. Should be one* "Information", "Warning", or "Error".* @return string Message severity*/public function getSeverity(){return $this->severity;}/*** Sets the severity of the message* @param $severity* @return void*/public function setSeverity($severity){$this->severity = $severity;}public function getMessage(){return $this->message;}public function setMessage($msg){$this->message = $msg;}}/** Counts the messages with the given severity in the array* of messages.** @param $messages An array of ResultMessage* @return int Count of messages with a severity of "Error"*/function countErrors($messages){$matchingCount = 0;foreach($messages as $m) {if ($m->getSeverity() == "Error") {$matchingCount++;}}return $matchingCount;}$messages = array(new ResultMessage("Error", "This is an error!"),new ResultMessage("Warning", "This is a warning!"),new ResultMessage("Error", "This is another error!"));$errs = countErrors($messages);echo("There are " . $errs . " errors in the result.\n");?>

 

处理错误

 根据大众的经验,如果要编写健壮的应用程序,错误处理要遵循 80/20 规则:80% 的代码用于处理异常和验证,20% 的代码用于完成实际工作。在编写程序的基本逻辑(happy-path)代码时经常这样做。这意味着编写适用于基本条件的代码,即所有的数据都是可用的,所有的条件符合预期。这样的代码在应用程序的生命周期中可能很脆弱。另一个极端是,甚至需要花大量时间为从未遇到过的条件编写代码。

  这一习惯要求您编写足够的错误处理代码,而不是编写对付所有错误的代码,以致代码迟迟不能完成。

不良习惯:根本没有错误处理代码

  清单 7 中的代码演示了两个不良习惯。第一,没有检查输入的参数,即使知道处于某些状态的参数会造成方法出现异常。第二,代码调用一个可能抛出异常的方法,但没有处理该异常。当发生问题时,代码的作者或维护该代码的人员只能猜测问题的根源。

清单 7. 不良习惯:不处理错误条件

 

良好习惯:处理异常

  清单 8 展示了以有意义的方式抛出和处理异常。额外的错误处理不仅使代码更加健壮,它还提高代码的可读性,使代码更容易理解。处理异常的方式很好地说明了原作者在编写方法时的意图。

清单 8. 良好习惯:处理异常

6) || ($day getMessage() . "\n");}try {echo("The name of the 'orange' day is: " . convertDayOfWeekToName('orange') . "\n");} catch (InvalidDayFormatException $e) {echo ("Encountered error while trying to convert value: " . $e->getMessage() . "\n");}?>

 

虽然检查参数是一种确认 — 如果您要求参数处于某种状态,这将对使用方法的人很有帮助 — 但是您应该检查它们并抛出有意义的异常:

  • 处理异常要尽量与出现的问题紧密相关。
  • 专门处理每个异常。

 

切忌使用复制粘贴

  您可以从其他地方将代码复制粘贴到自己的代码编辑器,但这样做有利也有弊。好的一面是,从一个示例或模板中复制代码能够避免很多错误。不好的一面是,这容易带来大量的类似编程方式。

  一定要注意,不要将代码从应用程序的一部分复制粘贴到另一部分。如果您采用这种方式,请停止这个不良的习惯,然后考虑将这段代码重写为可重用的。一般而言,将代码放置到一个地方便于日后的维护,因为这样只需在一个地方更改代码。

不良习惯:类似的代码段

  清单 9 给出了几个几乎一样的方法,只是其中的值不同而已。有一些工具可以帮助找到复制粘贴过来的代码(参见参考资料)。

清单 9. 不良习惯:类似的代码段

getSeverity() == "Error") {$matchingCount++;}}return $matchingCount;}/*** Counts the number of messages found in the array of* ResultMessage with the getSeverity() value of "Warning"** @param $messages An array of ResultMessage* @return unknown_type*/function countWarnings($messages){$matchingCount = 0;foreach($messages as $m) {if ($m->getSeverity() == "Warning") {$matchingCount++;}}return $matchingCount;}/*** Counts the number of messages found in the array of* ResultMessage with the getSeverity() value of "Information"** @param $messages An array of ResultMessage* @return unknown_type*/function countInformation($messages){$matchingCount = 0;foreach($messages as $m) {if ($m->getSeverity() == "Information") {$matchingCount++;}}return $matchingCount;}$messages = array(new ResultMessage("Error", "This is an error!"),new ResultMessage("Warning", "This is a warning!"),new ResultMessage("Error", "This is another error!"));$errs = countErrors($messages);echo("There are " . $errs . " errors in the result.\n");?>

 

良好习惯:带参数的可重用函数

  清单 10 展示了修改后的代码,它将复制的代码放到一个方法中。另一个方法也进行了更改,它现在将任务委托给新的方法。构建通用的方法需要花时间设计,并且这样做使您能停下来思考,而不是本能地使用复制粘贴。但有必要进行更改时,对通用的方法投入的时间将得到回报。

清单 10. 良好习惯:带参数的可重用函数

getSeverity() == $withSeverity) {$matchingCount++;}}return $matchingCount;}/*** Counts the number of messages found in the array of* ResultMessage with the getSeverity() value of "Error"** @param $messages An array of ResultMessage* @return unknown_type*/function countErrors($messages){return countMessages($messages, "Errors");}/*** Counts the number of messages found in the array of* ResultMessage with the getSeverity() value of "Warning"** @param $messages An array of ResultMessage* @return unknown_type*/function countWarnings($messages){return countMessages($messages, "Warning");}/*** Counts the number of messages found in the array of* ResultMessage with the getSeverity() value of "Warning"** @param $messages An array of ResultMessage* @return unknown_type*/function countInformation($messages){return countMessages($messages, "Information");}$messages = array(new ResultMessage("Error", "This is an error!"),new ResultMessage("Warning", "This is a warning!"),new ResultMessage("Error", "This is another error!"));$errs = countErrors($messages);echo("There are " . $errs . " errors in the result.\n");?>

 

结束语

  如果您在编写 PHP 代码的过程中养成本文讨论的良好习惯,您将能够构建易读、易理解、易维护的代码。使用这种方式构建的易维护代码将降低调试、修复和扩展代码所面临的风险。

  使用良好的名称和更短的方法能够提高代码的可读性。注释代码的目的有利于代码理解和扩展。适当地处理错误会使代码更加健壮。最后,停止使用复制粘贴,保持代码干净,提高可重用性。

PHP 相关文章推荐
PHP邮件专题
Oct 09 PHP
提升PHP执行速度全攻略
Oct 09 PHP
用PHP写的MySQL数据库用户认证系统代码
Mar 22 PHP
PHP 第二节 数据类型之字符串类型
Apr 28 PHP
给ECShop添加最新评论
Jan 07 PHP
php制作动态随机验证码
Feb 12 PHP
PHP结合Mysql数据库实现留言板功能
Mar 04 PHP
yii2 页面底部加载css和js的技巧
Apr 21 PHP
php日期操作技巧小结
Jun 25 PHP
php生成0~1随机小数的方法(必看)
Apr 05 PHP
使用Laravel中的查询构造器实现增删改查功能
Sep 03 PHP
thinkphp 获取控制器及控制器方法
Apr 16 PHP
PHP 截取字符串 分别适合GB2312和UTF8编码情况
Feb 12 #PHP
PHP 操作文件的一些FAQ总结
Feb 12 #PHP
php实现从ftp服务器上下载文件树到本地电脑的程序
Feb 10 #PHP
PHP6 mysql连接方式说明
Feb 09 #PHP
PHP 数据库树的遍历方法
Feb 06 #PHP
一周让你学会PHP 不错的学习资料
Feb 06 #PHP
简化php模板页面中分页代码的解析
Feb 06 #PHP
You might like
浅析php中jsonp的跨域实例
2013/06/21 PHP
php导出word格式数据的代码实例
2013/11/25 PHP
PHP7安装Redis扩展教程【Linux与Windows平台】
2016/09/30 PHP
Javascript学习笔记2 函数
2010/01/11 Javascript
Egret引擎开发指南之创建项目
2014/09/03 Javascript
javascript继承机制实例详解
2014/11/20 Javascript
jQuery的animate函数实现图文切换动画效果
2015/05/03 Javascript
对jQuary选择器的全面总结
2016/06/20 Javascript
JS实现颜色梯度与渐变效果完整实例
2016/12/30 Javascript
浅谈JS对html标签的属性的干预以及对CSS样式表属性的干预
2017/06/25 Javascript
原生JS进行前后端同构
2018/04/22 Javascript
解决JavaScript中0.1+0.2不等于0.3问题
2018/10/23 Javascript
vue插件mescroll.js实现移动端上拉加载和下拉刷新
2019/03/07 Javascript
typescript nodejs 依赖注入实现方法代码详解
2019/07/21 NodeJs
js针对图片加载失败的处理方法分析
2019/08/24 Javascript
Javascript摸拟自由落体与上抛运动原理与实现方法详解
2020/04/08 Javascript
Selenium执行JavaScript脚本的方法示例
2020/12/31 Javascript
在Python中使用元类的教程
2015/04/28 Python
Django小白教程之Django用户注册与登录
2016/04/22 Python
Python 爬取携程所有机票的实例代码
2018/06/11 Python
浅析python继承与多重继承
2018/09/13 Python
python文件选择对话框的操作方法
2019/06/27 Python
10款最好的Python开发编辑器
2019/07/03 Python
使用pandas 将DataFrame转化成dict
2019/12/10 Python
Pytorch1.5.1版本安装的方法步骤
2020/12/31 Python
python 装饰器重要在哪
2021/02/14 Python
全球领先的在线cosplay服装商店:RoleCosplay
2020/01/18 全球购物
用Python匹配HTML tag的时候,<.*>和<.*?>有什么区别
2012/11/04 面试题
个人自荐信
2013/12/05 职场文书
感恩寄语大全
2014/04/11 职场文书
2014年党务公开方案
2014/05/08 职场文书
合伙购房协议样本
2014/10/06 职场文书
股东授权委托书
2014/10/15 职场文书
Mysql 如何查询时间段交集
2021/06/08 MySQL
24年收藏2000多部退役军用电台
2022/02/18 无线电
SpringBoot项目部署到阿里云服务器的实现步骤
2022/06/28 Java/Android