您的位置:首页 > 编程语言 > PHP开发

什么样的PHP代码才算优秀, 外番“phper成长之道” 3ff8

2012-05-02 14:59 405 查看


优秀的PHP代码应该是结构化的。大段的代码应该被分割整理成一个个函数、方法,而那些不起眼的小段代码则应该加上注释,以便日后清楚它们的用途。而且应该尽可能地把前台代码如HTML、CSS、Javascript等从程序中分离出来。PHP的面向对象编程特性可以很好地帮助程序员将代码整理有序。

优秀的PHP代码应该是规范化的。无论是为变量名和函数名设定命名规则,还是对一些会重复使用的过程如数据库操作和错误处理进行标准化,抑或是简单到规定好代码是怎样缩进的,这些规范化都可以让代码的可读性大大提高。

优秀的PHP代码应该是自适应的。PHP有许多特性如magic quotes和short tags,这些特性的打开和关闭会影响到程序的运行。所以,一个好的程序员应该在他的代码中加如适当的语句来使程序能够根据环境进行调整。

优秀的PHP代码应该是安全的。虽然PHP是一种高效、灵活的语言,没有固定的框架,但却把安全问题留给了程序员们。对潜在安全漏洞的深刻理解,如跨站脚本攻击(XSS)、跨站请求伪造(CSRF)、代码注入漏洞、字符编码循环漏洞等,对于今天的专业程序员来说是至关重要的。

当应聘者在回答这些问题的时候,我就能清楚地知道是否该录用他。当然,有时程序员并不能很好地阐明这个问题,这时我们会让他们做一些PHP测试。测试中的许多问题表面上看起来非常简单,但这也给了应聘者们一个展现自我的机会,因为只要观察得仔细,就能找出问题。

下面这一小段“劣质”的PHP代码是一道简化了的测试题。这种问题就像在问:你该怎样优化这段代码?

<?
echo("<p>Search results for query: " .
$_GET['query'] . ".</p>");
?>

这段代码的主要问题在于它把用户提交的数据直接显示到了网页上,从而产生XSS漏洞。其实有很多方法可以填补这个漏洞。那么,什么代码是我们想要的呢?

<?
echo("<p>Search results for query: " .
htmlspecialchars($_GET['query']) . ".</p>");
?>

这是最低要求。XSS漏洞用htmlspecialchars函数填补了,从而屏蔽了非法字符。

<?php

if (isset($_GET['query']))
{
echo '<p>Search results for query: ',
htmlspecialchars($_GET['query'], ENT_QUOTES), '.</p>';
}

?>

能写出这样代码的人应该是我想要录用的人了。

<?被替换成了<?php,这样更符合XML规范。

在输出$_GET['query']的值之前先判断它是否为空。

echo命令中多余的括号被去掉了。

字符串用单引号限定,从而节省了PHP从字符串中搜索可替换的变量的时间。

用逗号代替句号,节省了echo的时间。

将ENT_QUOTES标识传递给htmlspecialchars函数,从而保证单引号也会被转义。虽然这并是最主要的,但也算是一个良好习惯。

----------------外一篇
40个迹象表明你还是PHP菜鸟
-------------

我愿意把本文归入我的“编程糗事”系列。尽管在正规大学课程中,接触到软件工程、企业级软件架构和数据库设计,但我还是时不时地体会到下述事实带给我的“罪恶”感,当然,都是我的主观感受,并且面向Eclipse:

你是PHP菜鸟,如果你:

不会利用如phpDoc这样的工具来恰当地注释你的代码

对优秀的集成开发环境如Zend Studio或Eclipse PDT视而不见

从未用过任何形式的版本控制系统,如Subclipse

不采用某种编码与命名标准,以及通用约定,不能在项目开发周期里贯彻落实

不使用统一开发方式

不转换(或)也不验证某些输入或SQL查询串(译注:参考PHP相关函数)

不在编码之前彻底规划你的程序

不使用测试驱动开发

不在错误开启状态下进行编码和测试(译注:参考PHP函数error_reporting)

对调试器的好处视而不见

不重构你的代码

不使用类似MVC模式把程序的不同层次划分开

不知道这些概念:KISS, DRY, MVC, OOP, REST

不用return而是直接在你的函数或类中输出(echo/print)内容

对单元测试或通用测试的优点视而不见

总是返回硬编码的HTML,却不返回纯粹的数据,字符串,或对象

总是对“消息”和“配置参数”进行硬编码

不对SQL查询语句做优化

不使用__autoload(译注:参考PHP手册相关描述)

不允许智能错误处理(译注:参考PEAR的ErrorStack)

使用$_GET替代$_POST来做具有破坏性的传递操作

不知道怎么利用正则表达式

从未听说过SQL注入或跨站脚本

不允许简易配置,也不允许类的构造函数接受参数传递而后执行set/get方法,或运行时的常量定义

不理解面向对象编程(OOP)的优势和劣势

不视情形大小而滥用OOP

自认为实现可复用的软件一定等于/需要让你的代码遵循OOP

不利用智能缺省值

没有单一的配置文件

不想暴露文件源码,却用.inc后缀名取代了.php

不使用数据库抽象层

不能保持DRY作风,即不重复自己,如果你总是在复制粘贴一些东西,说明你设计得很差劲

没有实现让一个函数/类/方法只做一件事,也不能组合利用它们

没能尝试OOP的特长,如抽象类、接口、多态、继承,访问控制修饰符(译注:如public, private, protected)

不用现有的设计模式优化你的程序体系设计

不允许你的用户在你拥有很多文件或目录的情况下定义基础目录

污染了名称空间,比如用常见字符串命名你的库函数

使用数据库表时不使用表前缀

不使用统一的模板引擎

不关注已有的PHP开发框架,懒于探索;其实先进的开发理念和美妙代码就蕴含其中。

and.... 如何突破PHP程序员的技术瓶颈

身边有几个做PHP开发的朋友,也接触到不少的PHP工程师,他们常疑虑自己将来在技术上的成长与发展,我常给他们一些建议,希望他们能破突自己,有更好的发展。

先明确我所指的PHP工程题,是指毕业工作后,主要以PHP进行WEB系统的开发,没有使用其的语言工作过。工作经验大概在3~4年,普通的WEB系统(百万级访问,千成级数据以内或业务逻辑不是特别复杂)开发起基本得心应手,没有什么问题。但他们会这样的物点:

除了PHP不使用其它的语言,可能会点shell 脚本。

对PHP的掌握不精(很多PHP手册都没有看完,库除外)

知识面比较窄(面对需求,除开使用PHP和mysql ,不知道其它的解决办法)

PHP代码以过程为主,认为面向对象的实现太绕,看不懂

这些PHPer 在遇到需要高性能,处理高并发,大量数据的项目或业务逻辑比较复杂(系统需要解决多领域业务的问题)时,缺少思路。不能分析问题的本质,技术判断力比较差,对于问题较快能找出临时的解决办法,但常常在不断临时性的解决办法中,系统和自己一步步走向崩溃。那怎么提高自己呢?怎么可以挑战难度更高的系统?

更高的挑战在那里?结合我自己的经验,我列出一些具体挑战,让大家先有个感性的认识。

高性能系统的挑战在那里?

如何选择WEB服务器?要不要使用fast-cgi 模式

要不要使用反向代理服务?选择全内存缓存还是硬盘缓存?

是否需要负载均衡?是基于应用层,还是网络层? 如何保证高可靠性?

你的PHP代码性能如何,使用优化工具后怎么样? 性能瓶颈在那里? 是否需要写成C的扩展?

用户访问有什么特点,是读多还是写多?是否需要读写分离?

数据如何存储?写入速度和读出速度如何? 数据增涨访问速读如何变化?

如何使用缓存? 怎么样考虑失效?数据的一致性怎么保证?

高复杂性系统的挑战在那里?

能否识别业务所对应的领域?是一个还是多个?

能否合理对业务进行抽象,在业务规则变化能以很小的代价实现?

数据的一致性、安全性可否保证?

是否撑握了面向对象的分析和设计的方法

当我所列出的问题,你都能肯定的回答,我想在技术上你基本已经可能成为架构师了。如何你还不能回答,你需要在以下几个方向加强。

如何你还不能回答,你需要在以下几个方向加强:

分析你所使用的技术其原理和背后运行的机制,这样可以提高你的技术判断力,提高你技术方案选择的正确性;

学习大学期间重要的知识, 操作系统原理,数据结构和算法。知道你以前学习都是为了考试,但现在你需要为自己学习,让自己知其所以然。

重新开始学习C语言,虽然你在大学已经学过。这不仅是因为你可能需要写PHP扩展,而且还因为,在做C的应用中,有一个时刻关心性能、内存控制、变量生命周期、数据结构和算法的环境。

学习面向对象的分析与设计,它是解决复杂问题的有效的方法。学习抽象,它是解决复杂问题的唯一之道。

"这么多的东西怎么学,这得学多久呀" ?如果你努力的话,有较好的规划,估计需要1~2年的时间。

如何有效的学习是一个大问题。 自己有些实践但很零散,不好总结。昨天晚上睡觉前,突然想到了RUP的核心,"以架构为中心,用例驱动,迭代开发",借用这个思想,关于有效的学习的方法,可以这样来表述:以原理、模型或机制为中心,任务驱动,迭代学习。

有点抽象, 举个例子来说明如何学习。目的: 学习如何提高处理性能。

可迭代驱动的任务: 通过IP找到所在地域。

这是WEB应用常见的任务,IP数据库是10左右万行的记录。

第一次迭代: 不考虑性能的情况下实现功能(通过PHP来实现)。因为无法直接通过KEY(IP)进行查找地域,所以直接放到数据或通过关联数组这种简单的方法都是不行的。思路还是先把数据进行排序,然后再进行查找。

如何通过IP查找? 已序的数据,二分查找是最快的。

如何排序?用库函数sort当然 是可以,但是即然是学习,那还是自己实现快速排序吧。

学习目标: 排序算法,查找算法。

PHPer 一般数据结构和算法基础比较差,平时也没有这方面的任务,自己也不学习,因此这方面的知识很缺乏。但是,编程解决的问题,最终都会归结到数据结构和对这种数据结构操作的算法。如果数据结构算法常在心中,那遇到问题就能清晰认识到它内在的结构,解决方法就会自然产生。

第二次迭代:优化数据的加载与排序。如果做到第一步,那基本上还是不可用,因为数据每次都需要的加载和排序,这样太耗时间。 解决的思路是,数据一次加载排序后,放到每个PHP进程能访问到的地方。

放到memcache 这是大家容易想到问题。其实放到共享内存(EA等加速器都支持)中是更快的方式,因为memcache还多了网络操作。 数据是整体放入到共享内存,还是分块放入,如何测试性能? 如何分析瓶颈所在(xdebug)? 在这些问题的驱动下你会学习到。

学习目标: 检测、定位、优化PHP性能的方法; PHP实现结构对性能的影响。
第三次迭代: 编写PHP的扩展。性能还是上不去,不得不进入C/C++的世界了,不过从此你将不只是PHPer 而服务端的全能型工程师,当然这对没有做过C/C++的同学挑战是巨大的。 我这里无法再简单来说如何学习C/C++ ,可以参看 《PHP程序员学习C++》

学习目标:C/C++的学习,PHP扩展的编写

怎么确定需要学习的机制和原理呢? 怎么找到驱动学习任务呢?我对需要学习的东西,都没有什么概念,怎么回答以上的两个问题?

从这个技术的定位来找出需要学习的重点,即它怎么做到(机制)的和它为什么能这样做到 (模型或原理)

列出这个技术最常见的应用,做为学习的任务,从简到难进行实践。

假如我需要学习Javascript ,我对于HTML,CSS有点感性认识。首要我了解到,JS 是WEB领域的动态语言,主要解决网页的动态交互的。那我要学习的要点如下:

JS如何与HTML 进行交互 (机制)

JS的动态特性在那里,与其它动态语言有何区别?(语言模型)

如果完全自学,找到需要学习的要点(机制、模型、原理) 设定学习任务的确不是那么容易把握。如果找到一个有经验的人来指导你或加一个学习型的团队,那学习的速度的确会大大提高。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: