阮一峰的网络日志 |
Posted: 26 Dec 2010 08:55 AM PST 今天下午,我在读下面这篇文章。 虽然名字叫《PHP最佳实践》,但是它主要谈的不是编程规则,而是PHP应用程序的合理架构。 它提供了一种逻辑和数据分离的架构模式,属于MVC模式的一种实践。我觉得,这是很有参考价值的学习资料,类似的文章网上并不多,所以一边学习,一边就把它翻译了出来。 根据自己的理解,我总结了它的MVC模式的实现方式(详细解释见译文):
======================================= PHP最佳实践 原载:http://www.odi.ch/prog/design/php/guide.php 译者:阮一峰
1. php.ini设置 php.ini控制了解释器的行为,下面的一些设置保证了你的程序有最大的可移植性。 i. short_open_tag 设为0,即永远使用PHP的长标签形式:<?php echo "hello world"; ?>,不用短标签形式<?= "hello world" ?>。 ii. asp_tags 设为0,不使用ASP标签<% echo "hello world"; %>。 iii. magic_quotes_gpc 建议在脚本中包含一个全局文件,负责在读取$_GET、$_POST、$_COOKIE变量之前,首先检查这个设置是否打开,如果打开了,这对这些变量应用stripslashes函数。(注:该设置已经在PHP 5.3中被废除。) iv. register_globals 不要依赖这个设置,永远通过全局变量$_GET、$_POST、$_COOKIE去读取GET、POST和COOKIE的值。为了方便起见,建议声明$PHP_SELF = $_SERVER['PHP_SELF']。 v. file_uploads 上传文件的最大大小,由下面的设置决定:
2. 配置文件(configuration file) 你应该把与应用程序相关的所有配置,写在一个文件里。这样你就能很方便地适应开发环境的变化。配置文件通常包含以下信息:数据库参数、email地址、各类选项、debug和logging输出开关、应用程序常数。 3. 名称空间(namespace) 选择类和函数名的时候,必须很小心,避免出现重名。尽可能不要在类以外,放置全局性函数,类对内部的属性和方法,相当于有一层名称空间保护。如果你确实有必要声明全局性函数,那么使用一个前缀,比如dao_factory()、 db_getConnection()、text_parseDate()等等。 4. 数据库抽象层 PHP不提供数据库操作的通用函数,每种数据库都有一套自己的函数。你不应该直接使用这些函数,否则一旦改用其他数据库(比如从MySQL 转为Oracle),你就有大麻烦了。而且,数据库抽象层通常比系统本身的数据库函数,更易用一些。 5. "值对象"(Value Object, VO) 值对象(VO)在形式上,就像C语言的struct结构。它是一个只包含属性、不包含任何方法(或只包含很少方法)的类。一个值对象,就对应一个实体。它的属性,通常应该与数据库的字段名保持相同。此外,还应该有一个ID属性。
6. 数据访问对象(Data Access Object, DAO) 数据访问对象(DAO)的作用,主要是将数据库访问与其他代码相隔离。DAO应该是可以叠加(stacked)的,这样就有利于将来你再添加数据库缓存。每一个值对象的类,都应该有自己的DAO。
DAO通常应该部署以下方法:
你可以根据自己的需要,添加其他DAO方法,常见的例子有isUsed()、getTop($n)、find($criteria)。 但是,所有的DAO方法都应该与数据库操作有关,不应该执行其他操作。DAO只应该对一张表进行基本的select / insert / update,不应该包含业务逻辑。举例来说,PersonDAO就不应该包含向某人发送Email的代码。 你可以写一个工厂函数,根据不同的类名,返回相应的DAO。
7. 自动生成代码 99%的值对象和DAO代码,可以根据数据库模式(schema)自动生成,前提是你的表和列使用约定的方式进行命名。如果你修改数据库模式,一个自动生成代码的脚本将大大节省你的时间。 8. 业务逻辑 业务逻辑直接反映使用者的需要。它们处理值对象,根据业务需要修改值对象的属性,使用DAO与数据库层交互。
9. 页逻辑(控制器) 当一个网页被请求时,页控制器(page controller)就会运行,然后产生输出。控制器的任务,就是将HTTP请求转化成业务对象(business object),然后调用相应的业务逻辑,最后生成一个"展示输出"的对象。 页逻辑依次执行以下步骤(请参照后面的PageController类的代码):
注意:可以编写一个工具函数(utility function),处理GET或POST值,当有的变量没有赋值时,提供一个默认值。页逻辑不包含HTML代码。
10. 表现层(Presentation Layer) 最顶层的页面包含实际的HTML代码。这个页面需要的所有业务对象(business object),由页逻辑提供。 这个页面先读取业务对象的属性,然后将它们转换成HTML格式。
11. 本地化(Localization) 本地化意味着要支持多种语言,这个比较麻烦,你无非有两种方法可以选择:
一般来说,A方法用得比较多,因为B方法会使得HTML页面的可读性很差。 所以,你可以先写完一种语言的页面,然后把它们进行拷贝,用某种命名法区别不同语言的版本,比如index_fr.php表示index.php的法语版。 为了保存用户的语言选择,你有几种方法:
看上去A方法比C方法容易得多(虽然session和cookie都有过期的问题),而B方法只能作为A或C的补充。 最后不要忘了,数据库中的字段也必须进行本地化。 12. 安装位置 有时候你需要知道程序的根目录在哪里,但是$_SERVER['DOCUMENT_ROOT']只是web服务器的根目录,如果你的程序安装在它的某个子目录之中,PHP没法自动知道。 你可以定义一个全局变量$ROOT,它的值就是程序的根目录,然后把它包含在每一个脚本文件中。那么,你要包含某个文件,就这样写require_once("$ROOT/lib/base.inc.php");。 13. 目录结构 首先,每个类都应该有自己的独立文件,还必须有一套文件名的命名规则(naming convention)。 软件的目录结构可以采用如下形式:
base.inc.php文件中,应该按照以下顺序添加包含文件:
至于那些存放图片、上传文件的目录,这里就省略了。 (完) |
You are subscribed to email updates from 阮一峰的网络日志 To stop receiving these emails, you may unsubscribe now. | Email delivery powered by Google |
Google Inc., 20 West Kinzie, Chicago IL USA 60610 |