7. 自动生成代码
99%的值对象和DAO代码,可以根据数据库模式schema)自动生成,前提是你的表和列使用约定的方式进行命名。如果你修改数据库模式,一个自动生成代码的脚本将大大节省你的时间。
8. 业务逻辑
业务逻辑直接反映使用者的需要。它们处理值对象,根据业务需要修改值对象的属性,使用DAO与数据库层交互。
- class NewsletterLogic {
- function NewsletterLogic() {
- ...
- }
- function subscribePerson(&$person) {
- ...
- }
- function unsubscribePerson(&$person) {
- ...
- }
- function sendNewsletter(&$newsletter) {
- ...
- }
- }
9. 页逻辑控制器)
当一个网页被请求时,页控制器page controller)就会运行,然后产生输出。控制器的任务,就是将HTTP请求转化成业务对象business object),然后调用相应的业务逻辑,最后生成一个"展示输出"的对象。
页逻辑依次执行以下步骤请参照后面的PageController类的代码):
◆假定页面请求之中,包含一个cmd参数。
◆根据cmd参数的值,执行相应的动作。
◆验证页面返回的值,生成一个值对象。
◆针对值对象,执行业务逻辑。
◆如果有必要,可以导向另一个页面。
◆收集必要的数据,输出结果。
注意:可以编写一个工具函数utility function),处理GET或POST值,当有的变量没有赋值时,提供一个默认值。页逻辑不包含HTML代码。
- class PageController {
- var $person; #$person is used by the HTML page
- var $errs;
- function PageController() {
- $action = Form::getParameter('cmd');
- $this->person = new Person();
- $this->errs = array();
- if ($action == 'save') {
- $this->parseForm();
- if (!this->validate()) return;
- NewsletterLogic::subscribe($this->person);
- header('Location: confirmation.php');
- exit;
- }
- }
- function parseForm() {
- $this->person->name = Form::getParameter('name');
- $this->person->birthdate = Util::parseDate(Form::getParameter('birthdate');
- ...
- }
- function validate() {
- if ($this->person->name == '') $this->errs['name'] = FORM_MISSING;
- #FORM_MISSING is a constant
- ...
- return (sizeof($this->errs) == 0);
- }
- }
10. 表现层
最顶层的页面包含实际的HTML代码。这个页面需要的所有业务对象business object),由页逻辑提供。这个页面先读取业务对象的属性,然后将它们转换成HTML格式。
- <?php
- require_once('control/ctl_person.inc.php'); #the page controller
- $c =& new PageController();
- ?>
- <html>
- <body>
- <form action="<?php echo htmlspecialchars($PHP_SELF) ?>" method="POST">
- <input type="hidden" name="cmd" value="save">
- <input type="text" name="name"
- value="<?php echo htmlspecialchars($c->person->name); ?>">
- <button type="submit">Subscribe</button>
- </form>
- </body>
- </html>
11. 本地化Localization)
本地化意味着要支持多种语言,这个比较麻烦,你无非有两种方法可以选择:
A) 准备多重页面。
B) HTML页面中去除特定语言相关的内容。
一般来说,A方法用得比较多,因为B方法会使得HTML页面的可读性很差。所以,你可以先写完一种语言的页面,然后把它们进行拷贝,用某种命名法区别不同语言的版本,比如index_fr.php表示index.php的法语版。为了保存用户的语言选择,你有几种方法:
A) 将语言设定保存在一个session变量或cookie之中;
B) 从HTTP头中读取locale值;
C) 把语言设定作为一个参数,追加在每个URL后面。
看上去A方法比C方法容易得多虽然session和cookie都有过期的问题),而B方法只能作为A或C的补充。最后不要忘了,数据库中的字段也必须进行本地化。
12. 安装位置
有时候你需要知道程序的根目录在哪里,但是$_SERVER['DOCUMENT_ROOT']只是web服务器的根目录,如果你的程序安装在它的某个子目录之中,PHP没法自动知道。
你可以定义一个全局变量$ROOT,它的值就是程序的根目录,然后把它包含在每一个脚本文件中。那么,你要包含某个文件,就这样写require_once("$ROOT/lib/base.inc.php");。
13. 目录结构
首先,每个类都应该有自己的独立文件,还必须有一套文件名的命名规则naming convention)。软件的目录结构可以采用如下形式:
- / 根目录。浏览器从这个页面开始访问。
- /lib/ 包含全局变量base.inc.php)和配置文件config.inc.php)。
- /lib/common/ 包含其他项目也可以共用的库,比如数据库抽象层。
- /lib/model/ 包含值对象类。
- /lib/dao/ 包含数据访问对象DAO)类,以及DAO工厂函数。
- /lib/logic/ 包含业务逻辑类。
- /parts/ 包含HTML模板文件。
- /control/ 包含页逻辑。对于大型程序来说,这个目录下面可能还有子目录比如admin/, /pub/)。
base.inc.php文件中,应该按照以下顺序添加包含文件:
- * /lib/common之中经常使用的类比如数据库层)。
- * 配置文件;
- * /lib/model之中所有类;
- * /lib/dao的之中所有类。
至于那些存放图片、上传文件的目录,这里就省略了。
原文地址:http://www.ruanyifeng.com/blog/2010/12/php_best_practices.html
PHP之友评论