时间:2021-07-01 10:21:17 帮助过:23人阅读
PHP 编程规范
1. 命名规范
1.1变量
1.1.1全局变量
全局变量使用$g_开头,如$g_data_list。
1.1.2 一般变量
一般的变量使用小写字母命名,单词之间使用下划线分隔。
变量名字应该使用名词或者形容词+名词的方式。如$value,$new_value。
1.1.3 临时变量
不要将在循环中频繁使用的临时变量如$i,$j等用于其它用途。
1.2 函数
函数名用驼峰式命名,第一个单词的首字母小写,禁用下划线等连接符。
函数的命名建议使用动词+名词的方式,如getUserImg()。
完成一组功能的函数放到一个文件中,存放函数的文件采用name.func.php命名。
1.3 类
类使用驼峰式命名,英文的大小写来分隔单词,包括首个单词,所有单词的首字母大写,如PageManager;
一般情况下,一个类对应到一个文件;
当一些类关系紧密时,可以存放在一个文件中;
存放类的文件采用ClassName.class.php方式命名。
1.4 方法
方法使用英文的大小写来分隔单词,除首个单词外,其他单词的首字母大写,如getCurrentPage();
不要采用不常用的缩写,如where2go();
使用常用的缩写时,只大写首字母,如getHtml()。
2. 版式规则
2.1 语义分隔
各个函数、方法之间应该采用空行间隔;
同一个函数中联系紧密的语句之间可以不换行,其他情况需要换行。
2.2 空格规则
2.2.1 逻辑运算符前后必须加空格
正确: $a == $b;
错误: $a==$b;
$a ==$b;
正确: $a++; $a--;
错误: $a ++; $a --;
备注: 加一减一运算符不能加空格。
2.2.2 多个参数分隔时必须加空格
正确: $g_pro , $g_user , g_show;
getDbInfo($host, $user, $passwd);
错误: $g_pro,$g_user,$g_show;
getDbInfo($host,$user,$passwd);
2.2.3 语法关键字后必须加空格
例如:If, for , while, switch …..
正确:for ($i = 0; $i < 10; $i++)
错误:for($i = 0; $i < 10; $i++ )
2.3 字符串和变量连接规则
字符串与变量连接使用'.'号时,必须在'.'前后加空格,使用"号时,必须在变量前后加"{}"。
正确:$my_name = 'file_' . $var1;
$my_name = "file_{$var1}";
错误:$my_name = "file_'.$var1;
$my_name = "file_$var1";
2.4 圆括号规则
函数名后括号不需要加空格、语法关键字后的括号必须加空格。
正确:for ($i = 0; $i < 10; $i++)
strlen($my_name);
错误:for($i = 0; $i < 10; $i++ )
strlen ($my_name);
2.5 花括号规则
花括号必须为上下对应。
正确:
if ($a)
{
$b = $a;
}
错误:
if ($a){
$b = $a;
}
2.6 数组定义规则
数组定义和使用时中key值前后必须加单引号。
PHP 代码:
正确:
$user_info = array(
'name' => '',
'gender' => ''
);
echo $user_info['name'];
错误:
$user_info = array(
name => '',
gender => ''
);
echo $user_info[name];
2.7 SQL规则
在PHP中嵌入的SQL语句关键字全部采用大写;
表名和字段名要用反引号(`)引起来以防止因为字段名中包含空格而出现错误;
数据值两边用单引号''包括,并且应确保数据值中的单引号已经转义以防止SQL注入。
正确: $sql = "SELECT `user`.`name` FROM `user` WHERE `id` = '{$id}' LIMIT 1";
错误: $sql = "select name.user from name where id = {$id} ";
备注: Sqlite 不支持‘`’, 单引号转义是单引号前加单引号如:”’’a”
3. 注释规则
3.1 一般规则
不写不必要的注释;只有当代码不能很好地说明逻辑时,才用注释补充;
把注释看成程序的一部分,在编写/维护代码时同时编写/维护注释;
注释完全采用PHPDocumentor的规范,以方便用其生成API级文档。
3.2 详细规则
请参见PHPDocumentor手册。下边给出各个部分的注释示范。
3.2.1 版权信息
注释名称 版权信息
注释示范 //
// +----------------------------------------------------+
// | phpDocumentor |
// +----------------------------------------------------+
// | Copyright (c) 2000-2003 Joshua Eichorn |
// | Email jeichorn@phpdoc.org |
// | Web http://www.phpdoc.org |
// +----------------------------------------------------+
// | This source file is subject to PHP License |
// +----------------------------------------------------+
//
备注: 使用//来标示版权信息,以免和PHPDocumentor的page-level DocBlock发生冲突
3.2.2文件头注释示例
注释名称 文件头注释
注释示范
PHP 代码:
/**
* All abstract representations of inline tags are in this file
* @package phpDocumentor
* @subpackage InlineTags
* @since separate file since version 1.2
* @version $Id $
*/
备注:
1 文件头注释需要指明所属的包和子包
2 在@version中加上$ID,以方便使用CVS管理文件
3.2.3 类注释示例
注释名称 类注释
注释示范
PHP 代码:
/**
* Use this element to represent an {@}inline tag} like {@}link}
* @see parserStringWithInlineTags
* @package phpDocumentor
* @subpackage InlineTags
* @author Greg Beaver
* @since 1.0rc1
* @version $Revision: 1.21.2.6 $
* @tutorial inlinetags.pkg
*/
3.2.4 类属性注释示例
注释名称 类属性注释
注释示范
PHP 代码:
/**
* Element type
*
* Type is used by many functions to skip the hassle of
*
*
* if get_class($blah) == 'parserBlah'
*
* always "inlinetag"
* @var string
*/
var $type = 'inlinetag';
3.2.5 函数/类方法注释示例
注释名称 函数/类方法注释
注释示范
PHP 代码:
/**
* @return string always ''
* calculate the short description of a DocBlock
* @see parserStringWithInlineTags::getString()
* @see parserStringWithInlineTags::trimmedStrlen()
*/
function getString()
{
return '';
}
4. 命名规则
4.1 类命名
在为类(class)命名前首先要知道它是什么。如果通过类名的提供的线索,你还是想不起这个类是什么的话,那么你的设计就还做的不够好。
超过三个词组成的混合名是容易造成系统各个实体间的混淆,再看看你的设计,尝试使用(CRC Session card)看看该命名所对应的实体是否有着那么多的功用。
对于派生类的命名应该避免带其父类名的诱惑,一个类的名字只与它自身有关,和它的父类叫什么无关。
有时后缀名是有用的,例如:如果你的系统使用了代理(agent),那么就把某个部件命名为“下载代理”(DownloadAgent)用以真正的传送信息。
4.2 方法和函数命名
通常每个方法和函数都是执行一个动作的,所以对它们的命名应该清楚的说明它们是做什么的:用checkForErrors()代替errorCheck(),用dumpDataToFile()代替dataFile()。这么做也可以使功能和数据成为更可区分的物体。
有时后缀名是有用的:
Max - 含义为某实体所能赋予的最大值。
Cnt - 一个运行中的计数变量的当前值。
Key - 键值。
例如:RetryMax 表示最多重试次数,RetryCnt 表示当前重试次数。
有时前缀名是有用的:
Is - 含义为问一个关于某样事物的问题。无论何时,当人们看到Is就会知道这是一个问题。
Get - 含义为取得一个数值。
Set - 含义为设定一个数值
例如:IsHitRetryLimit。
4.3 缩写词不要全部使用大写字母
无论如何,当遇到以下情况,你可以用首字母大写其余字母小写来代替全部使用大写字母的方法来表示缩写词。
使用: getHtmlStatistic.
不使用: getHTMLStatistic.
理由:
当命名含有缩略词时,人们似乎有着非常不同的直觉。统一规定是最好,这样一来,命名的含义就完全可以预知了。
举个networkABCKey的例子,注意C是应该是ABC里面的C还是key里面的C,这个是很令人费解的。有些人不在意这些,其他人却很讨厌这样。所以你会在不同的代码里看到不同的规则,使得你不知道怎么去叫它。
例如:
class FluidOz // 不要写成 FluidOZ
class GetHtmlStatistic // 不要写成 GetHTMLStatistic
4.4 类命名
使用大写字母作为词的分隔,其他的字母均使用小写
名字的首字母使用大写
不要使用下划线('_')
理由:
根据很多的命名方式,大部分人认为这样是最好的方式。
例如:
class NameOneTwo
class Name
4.5 类库命名
目前命名空间正在越来越广泛的被采用,以避免不同厂商和团体类库间的类名冲突。
当尚未采用命名空间的时候,为了避免类名冲突,一般的做法是在类名前加上独特的前缀,两个字符就可以了,当然多用一些会更好。
例如:
John Johnson的数据结构类库可以用Jj做为前缀,如下:
class JjLinkList
{
}
另一种折中方式是建立包含类库目录(事实上Java也是这么做的),以不通的目录代表不同的命名空间。
例如:
Microsoft的数据库相关类库可以在:
/classes/com/Microsoft/ Database/DbConn.php
Apache的数据库相关类库可在:
/classes/org/apache/Database/DbConn.php
4.6 方法命名
函数名只能包含字母和数字字符,不允许使用下划线,虽然允许使用数字,但并不提倡。
函数名必须以小写字母开始。如果函数名包含多个单词,那么除了第一个单词,其他单词的首字母都大写,这就是通常所称的骆驼法则("studlyCaps"或"camelCaps")。
为增强代码的可读性,允许命名一定程度的详细和冗长。
下面是一些合乎规范的函数命名的例子:
filterInput()
getElementById()
widgetFactory()
对于面向对象编程,对象的存取器总是以"get"或"set"为前缀;当使用设计模式时,比如singleton或工厂模式(factory),方法的命名通常会包含模式的名称,这样可增强代码可读性,并且能一目了然的知道采用了何种设计模式。
本规范不允许有全局范围的函数(即游离于对象之外的函数),这些函数应该包装进一个静态类。
4.7 类属性命名
属性命名应该以字符‘m’为前缀。
前缀‘m’后采用于类命名一致的规则。
‘m’总是在名字的开头起修饰作用,就像以‘r’开头表示引用一样。
理由:
前缀'm'防止类属性和方法名发生任何冲突。你的方法名和属性名经常会很类似,特别是存取元素。
例如:
class NameOneTwo
{
function varAbc() {};
function errorNumber() {};
var $mVarAbc;
var $mErrorNumber;
var $mrName;
}
4.8 方法中参数命名
第一个字符使用小写字母。
在首字符后的所有字都按照类命名规则首字符大写。
理由:
可以区分方法中的一般变量。
你可以使用与类名相似的名称而不至于产生重名冲突。
例如:
class NameOneTwo
{
function startYourEngines(
&$rSomeEngine,
&$rAnotherEngine);
}
4.9 变量命名
所有字母都使用小写
使用'_'作为每个词的分界。
理由:
通过这一途径,代码中变量的作用域是清晰的。
所有的变量在代码中都看起来不同,容易辨认。
例如
function handleError($errorNumber)
{
$error = OsErr($errorNumber);
$time_of_error = OsErr->getTimeOfError();
$error_processor = OsErr->getErrorProcessor();
}
4.10引用变量和函数返回引用
引用必须带‘r’前缀
理由:
使得类型不同的变量容易辨认
它可以确定哪个方法返回可更改对象,哪个方法返回不可更改对象。
例如:
class Test
{
var mrStatus;
function doSomething(&$rStatus) {};
function &rStatus() {};
}
4.11 全局变量
全局变量应该带前缀‘g’。
理由
知道一个变量的作用域是非常重要的。
例如:
global $gLog;
global &$grLog;
4.12 定义命名 / 全局常量
全局常量用'_'分隔每个单词。
理由
这是命名全局常量的传统。你要注意不要与其它的定义相冲突。
例如:
define("A_GLOBAL_CONSTANT", "Hello world!");
4.13 静态变量
静态变量应该带前缀‘s’。
理由:
知道一个变量的作用域是非常重要的。
例如:
function test()
{
static $msStatus = 0;
}
4.14 函数命名
函数名字采用CGNU的惯例,所有的字母使用小写字母,使用'_'分割单词。
理由:
这样可以更易于区分相关联的类名。
例如:
function some_bloody_function()
{
}
4.15错误返回检测规则
检查所有的系统调用的错误信息,除非你要忽略错误。
为每条系统错误消息定义好系统错误文本以便include。
5.书写规则
5.1 大括号 {} 规则
在三种主要的大括号放置规则中,有两种是可以接受的,如下的第一种是最好的:
将大括号放置在关键词下方的同列处:
if ($condition) while ($condition)
{ {
... ...
} }
传统的UNIX的括号规则是,首括号与关键词同行,尾括号与关键字同列:
if ($condition) { while ($condition) {
... ...
} }
理由:
引起剧烈争论的非原则的问题可通过折衷的办法解决,两种方法任意一种都是可以接受的,然而对于大多数人来说更喜欢第一种。原因就是心理研究学习范畴的东西了。
对于更喜欢第一种还有着更多的原因。如果您使用的字符编辑器支持括号匹配功能的话(例如vi),最重要的就是有一个好的样式。为什么?我们说当你有一大块的 程序而且想知道这一大块程序是在哪儿结束的话。你先移到开始的括号,按下按钮编辑器就会找到与之对应的结束括号,例如:
if ($very_long_condition && $second_very_long_condition)
{
...
}
else if (...)
{
...
}
从一个程序块移动到另一个程序块只需要用光标和你的括号匹配键就可以了,不需找匹配的括号。
5.2 缩进/制表符/空格 规则
使用制表符缩进。
使用三到四个空格为每层次缩进。
不再使用只要一有需要就缩排的方法。对于最大缩进层数,并没有一个固定的规矩,假如缩进层数大于四或者五层的时候,你可以考虑着将代码因数分解(factoring out code)。
理由:
许多编程者支持制表符。
当人们使用差异太大的制表符标准的话,会使阅读代码变得很费力。
如此多的人愿意限定最大的缩进层数,它通常从未被看作是一件工作。我们相信程序员们会明智的选择嵌套的深度。
例如:
function func()
{
if (something bad)
{
if (another thing bad)
{
while (more input)
{
}
}
}
}
5.3 小括号、关键词和函数规则
不要把小括号和关键词紧贴在一起,要用空格隔开它们。
不要把小括号和函数名紧贴在一起。
除非必要,不要在Return返回语句中使用小括号。
理由:
关键字不是函数。如果小括号紧贴着函数名和关键字,二者很容易被看成是一体的。
例如:
if (condition)
{
}
while (condition)
{
}
strcmp($s, $s1);
return 1;
5.4 别在对象架构函数中做实际的工作
别在对象架构构造函数中做实际的工作,构造函数应该包含变量的初始化和(或)不会发生失败的操作。
理由:
构造不能返回错误 。
例如:
class Device
{
function Device() { /* initialize and other stuff */ }
function Open() { return FAIL; }
};
$dev = new Device;
if (FAIL == $dev->Open()) exit(1);
5.5 If Then Else 格式
布局
这由程序员决定。不同的花括号样式会产生些微不同的样观。一个通用方式是:
if (条件1) // 注释
{
}
else if (条件2) // 注释
{
}
else // 注释
{
}
如果你有用到else if 语句的话,通常最好有一个else块以用于处理未处理到的其他情况。可以的话放一个记录信息注释在else处,即使在else没有任何的动作。
条件格式
总是将恒量放在等号,不等号的左边,例如:
if ( 6 == $errorNum ) ...
一个原因是假如你在等式中漏了一个等号,语法检查器会为你报错。第二个原因是你能立刻找到数值而不是在你的表达式的末端找到它。需要一点时间来习惯这个格式,但是它确实很有用。
5.6 switch 格式
当一个case块处理后,直接转到下一个case块处理,在这个case块的最后应该加上注释。
default case总应该存在,它应该不被到达,然而如果到达了就会触发一个错误。
如果你要创立一个变量,那就把所有的代码放在块中。
例如:
switch (...)
{
case 1:
...
// FALL THROUGH
case 2:
{
$v = get_week_number();
...
}
break;
default:
}
5.7 continue,break 和 ? 的使用
5.7.1 Continue 和 Break
Continue 和 break 其实是变相的隐蔽的 goto方法。
Continue 和 break 像 goto 一样,它们在代码中是有魔力的,所以要节俭(尽可能少)的使用它们。使用了这一简单的魔法,由于一些未公开的原因,读者将会被定向到只有上帝才知道的地方去。
Continue有两个主要的问题:
它可以绕过测试条件。
它可以绕过不等表达式。
看看下面的例子,考虑一下问题都在哪儿发生:
while (TRUE)
{
...
// A lot of code
...
if (/* some condition */)
{
continue;
}
...
// A lot of code
...
if ( $i++ > STOP_VALUE) break;
}
注意:"A lot of code"是必须的,这是为了让程序员们不能那么容易的找出错误。
通过以上的例子,我们可以得出更进一步的规则:continue 和 break 混合使用是引起灾难的正确方法。
5.7.2 ?:
麻烦在于人们往往试着在 ? 和 : 之间塞满了许多的代码。以下的是一些清晰的连接规则:
把条件放在括号内以使它和其他的代码相分离。
如果可能的话,动作可以用简单的函数。
把所做的动作,“?”,“:”放在不同的行,除非他们可以清楚的放在同一行。
例如:
(condition) ? funct1() : func2();
or
(condition)
? long statement
: another long statement;
5.8 声明块的定位
声明代码块需要对齐。
理由:
清晰。
变量初始化的类似代码块应该列表。
&应靠近类型,而不是变量名。
例如:
var $mDate
var& $mrDate
var& $mrName
var $mName
$mDate = 0;
$mrDate = NULL;
$mrName = 0;
$mName = NULL;
5.9 每行一个语句
除非这些语句有很密切的联系,否则每行只写一个语句。
5.10 短方法
方法代码要限制在一页内。
5.11 记录所有的空语句
总是记录下for或者是while的空块语句,以便清楚的知道该段代码是漏掉了,还是故意不写的。
while ($dest++ = $src++)
; // VOID
5.12 不要采用缺省方法测试非零值
不要采用缺省值测试非零值,也就是使用:
if (FAIL != f())
比下面的方法好:
if (f())
即使 FAIL 可以含有 0 值 ,也就是PHP认为false的表示。在某人决定用-1代替0作为失败返回值的时候,一个显式的测试就可以帮助你了。就算是比较值不会变化也应该使用显式的比较;例如:if (!($bufsize % strlen($str)))应该写成:if (($bufsize % strlen($str)) == 0)以表示测试的数值(不是布尔)型。一个经常出问题的地方就是使用strcmp来测试一个字符等式,结果永远也不会等于缺省值。
非零测试采用基于缺省值的做法,那么其他函数或表达式就会受到以下的限制:
只能返回0表示失败,不能为有其他的值。
命名以便让一个真(true)的返回值是绝对显然的,调用函数IsValid()而不是Checkvalid()。
5.13 布尔逻辑类型
大部分函数在FALSE的时候返回0,但是发挥非0值就代表TRUE,因而不要用1(TRUE,YES,诸如此类)等式检测一个布尔值,应该用0(FALSE,NO,诸如此类)的不等式来代替:
if (TRUE == func()) { ...
应该写成:
if (FALSE != func()) { ...
5.14 通常避免嵌入式的赋值
有时候在某些地方我们可以看到嵌入式赋值的语句,那些结构不是一个比较好的少冗余,可读性强的方法。
while ($a != ($c = getchar()))
{
process the character
}
++和--操作符类似于赋值语句。因此,出于许多的目的,在使用函数的时候会产生副作用。使用嵌入式赋值提高运行时性能是可能的。无论怎样,程序员在使用嵌入式赋值语句时需要考虑在增长的速度和减少的可维护性两者间加以权衡。例如:
a = b + c;
d = a + r;
不要写成:
d = (a = b + c) + r;
虽然后者可以节省一个周期。但在长远来看,随着程序的维护费用渐渐增长,程序的编写者对代码渐渐遗忘,就会减少在成熟期的最优化所得。
6. 帮助与共享
6.1 重用您和其他人的艰苦工作
跨工程的重用在没有一个通用结构的情况下几乎是不可能的。对象符合他们现有的服务需求,不同的过程有着不同的服务需求环境,这使对象重用变得很困难。
开发一个通用结构需要预先花费许多的努力来设计。当努力不成功的时候,无论出于什么原因,有几种办法推荐使用:
6.2 请教!给群组发Email求助
这个简单的方法很少被使用。因为有些程序员们觉得如果他向其他人求助,会显得自己水平低,这多傻啊!做新的有趣的工作,不要一遍又一遍的做别人已经做过的东西。
如果你需要某些事项的源代码,如果已经有某人做过的话,就向群组发email求助。结果会很惊喜哦!
在许多大的群组中,个人往往不知道其他人在干什么。你甚至可以发现某人在找一些东西做,并且自愿为你写代码,如果人们在一起工作,外面就总有一个金矿。
6.3 告诉!当你在做事的时候,把它告诉所有人
如果你做了什么可重用的东西的话,让其他人知道。别害羞,也不要为了保护自豪感而把你的工作成果藏起来。一旦养成共享工作成果的习惯,每个人都会获得更多。
6.4 小型代码库
对于代码重用,一个常见的问题就是人们不从他们做过的代码中做库。一个可重用的类可能正隐蔽在一个程序目录并且决不会有被分享的激动,因为程序员不会把类分拆出来加入库中。
这样的其中一个原因就是人们不喜欢做一个小库,对小库有一些不正确感觉。把这样的感觉克服掉吧,电脑才不关心你有多少个库呢。
如果你有一些代码可以重用,而且不能放入一个已经存在的库中,那么就做一个新的库吧。如果人们真的考虑重用的话,库不会在很长的一段时间里保持那么小的。
6.5 知识库
很多公司不清楚现有什么代码可用,而且大多数程序员仍然没有通过沟通他们已经做过了什么,或者一直在询问现存什么代码可用。解决这个的方法是有一个可用的知识库。
理想的情况是,程序员可以到一个WEB页,浏览或者查询打包的知识库列表,找到他们所要的。建立一个程序员可以自动维护的知识库系统,是一个很不错的做法。如果有一个专门的管理员来负责维护这个知识库,那当然更好。
另一种方法是自动的从代码中产生知识库的做法。把通用的类、方法和标头(subsystem headers)作为手册或者是知识库的一个条目。
7. 书写注释
7.1 讲一个故事
把你的注释当作描述系统的一个故事。并且使得你的注释能被机器解析后,以固定的格式放到手册中去。类的注释是故事的一部分,方法的名称、方法的注释、方法的实现也是故事一部分。所有的这些部分编织在一起,使得人们在以后的时间里能够准确的知道你干了什么,为什么这么做。
7.2 归档注释
注释的要归档才有意义,否则,假如在一个地方放一条注释描述你做了什么选择和你为什么这么做,只有考古学家才能发现这是最有用的信息。(如何归档另行规范)
7.3注释结构
工程的每部分都有特定的注释结构。 程序中的注释,这里给出示例作为规范,注释中以 * @ 为关键字的开始,以:为注释关键字结尾。
7.3.1 预定义关键字
关键字 含义
Purpose 表示类、属性、方法要做些什么或者什么含义。
Package Name 类名
Author 作者
Modifications 修改记录(编号规则为“No”+日期+“-”+序号)
See 参考
Method Name 方法名
Parameter 参数名(包括类型)
Return 返回值(包括类型)
Attribute/Variable Name 属性/变量名
Type 属性/变量类型
7.3.2 类的注释
/**
* @ Purpose:
* 访问数据库的类,以ODBC作为通用访问接口
* @Package Name: Database
* @Author: Forrest Gump gump@crtvu.edu.cn
* @Modifications:
* No20020523-100:
* odbc_fetch_into()参数位置第二和第三个位置调换
* John Johnson John@crtvu.edu.cn
* @See: (参照)
*/
class Database
{
……
}
7.3.3 方法注释
/**
* @Purpose:
* 执行一次查询
* @Method Name: Query()
* @Parameter: string $queryStr SQL查询字符串
* @Return: mixed 查询返回值(结果集对象)
*/
function($queryStr)
{
……
}
7.3.4 属性或变量注释
/**
* @Purpose:
* 数据库连接用户名
* @Attribute/Variable Name: mDbUserName
* @Type: string
*/
var mDbUserName;
7.3.5 if (0)来注释外部代码块
有时需要注释大段的测试代码,最简单的方法就是使用if (0)块:
function example()
{
great looking code
if (0)
{
lots of code
}
more code
}
你不能使用/**/,因为注释内部不能包含注释,而大段的程序中可以包含注释。
7.3.6 目录文档
所有的目录下都需要具有README文档,其中包括:
该目录的功能及其包含内容
一个对每一文件的在线说明(带有link),每一个说明通常还应该提取文件标头的一些属性名字。
包括设置、使用说明
指导人们如何连接相关资源:
源文件索引
在线文档
纸文档
设计文档
其他对读者有帮助的东西
考虑一下,当每个原有的工程人员走了,在6个月之内来的一个新人,那个孤独受惊吓的探险者通过整个工程的源代码目录树,阅读说明文件,源文件的标头说明等等做为地图,他应该有能力穿越整个工程。
8. 其他
8.1 采用面向对象的设计方法;
理由:
毫无疑问这是最接近人们自然思维的方法,可能前期会觉得没有直接书写来得快,能否试着保留自己的看法?好戏在后头!
8.2 类的定义采用一个文件一个类,并且类名和文件名相同;
理由:
越来越多的人接受了这种做法
事实证明这种方法使得项目的逻辑结构更清晰
8.3 类定义文件中,定义体之外不得出现诸如echo、print等输出语句;
理由:
出现这样的语句,应该当做出现bug来看。
8.4 输出网页的页面不出现SQL语句
理由:
这是n层结构的编程思想所致,每层的任务不同,虽然可以越权行使,可能这样很快捷,但我们不赞成这么干。
8.5 进行SQL执行的数据必须进行有效性检测
8.5.1 特殊符号:
对于MS SQL Server,’%_[ ] 这些符号都是在书写SQL语句中的特殊含义字符,在SQL执行前需要对这些字符进行处理。
8.5.2脚本符号:
对于PHP脚本标记,如?><%%>,在进入数据库前需要检测处理。
理由:
这是数据库编程的一个约定,很多参考书上也是这么说,这里需要强调一下。
8.6 在HTML网页中尽量不要穿插PHP代码
循环代码和纯粹变量输出(类似于=$UserName?>)除外。
理由:
需要说明的是我们工作的上游,页面设计者的工作,假如在页面中穿插代码,将破坏结构,这应当是我们需要避免的。
在这里的PHP代码只负责显示,多余的代码显然是不应该的。
8.7没有含义的数字
一个在源代码中使用了的赤裸裸的数字是不可思议的数字,因为包括作者,在三个月内,没人它的含义。例如:
if (22 == $foo)
{
start_thermo_nuclear_war();
}
else if (19 == $foo)
{
refund_lotso_money();
}
else if (16 == $foo)
{
infinite_loop();
}
else
{
cry_cause_im_lost();
}
在上例中22和19的含义是什么呢?如果一个数字改变了,或者这些数字只是简单的错误,你会怎么想?
使用不可思议的数字是该程序员是业余运动员的重要标志.
你应该用define()来给你想表示某样东西的数值一个真正的名字,而不是采用赤裸裸的数字,例如:
define("PRESIDENT_WENT_CRAZY", "22");
define("WE_GOOFED", "19");
define("THEY_DIDNT_PAY", "16");
if (PRESIDENT_WENT_CRAZY == $foo)
{
start_thermo_nuclear_war();
}
else if (WE_GOOFED == $foo)
{
refund_lotso_money();
}
else if (THEY_DIDNT_PAY == $foo)
{
infinite_loop();
}
else
{
happy_days_i_know_why_im_here();
}
现在不是变得更好了么?
9. PHP文件扩展名
常见的PHP文件的扩展名有:html, .php, .php3, .php4, .phtml, .inc, .class...
这里我们约定:
所有浏览者可见页面使用.html
所有类、函数库文件使用.php
理由:
扩展名描述的是那种数据是用户将会收到的。PHP是解释为HTML的。
10. PHP代码标记
统一使用,禁用短标签。