当前位置:  开发笔记 > 编程语言 > 正文

一步步编写PHP的Framework(十五)

从今天开始我们开始介绍模型,模型是一个框架中非常重要的一块儿,控制器实际上不能进行数据的处理,这种处理的过程我们全部放在模型这一块儿来做。如果是Java,即使是没用框架,它的模型这一块儿也挺复...">

 

从今天开始我们开始介绍模型,模型是一个框架中非常重要的一块儿,控制器实际上不能进行数据的处理,这种处理的过程我们全部放在模型这一块儿来做。

        如果是Java,即使是没用框架,它的模型这一块儿也挺复杂的,首先编写service接口,然后写实现,然后定义DAO接口,DAO实现,然后定义这几层之间传输数据的Domain,一般我认为它就是一个POJO。

        Java这种契约式的编程方式很不错,即使后面的实现类发生改变,接口没有改变,代码的修改的量也不是很多。

        在PHP这一块儿,相对而言就要弱一些,我们如果将这个抽象的太深,那么效率必然会收到影响,抽象的比较浅,模型这一层后期维护又是一个问题。

        首先我们先说一下最常用的方式,访问DB,如果只是着眼于单个表,那么就存在表模型,一个表就是一个模型,我们称为DbTable;如果着眼于表与表之间的关联,那么就存在关系模型,我们称为DbRelation。

        由于DbTable只与单个表有关,所以它的操作比较简单,而DbRelation由于处理的是表与表之间的关系,所以比较复杂。

        现在我们举一个例子,一个系统有三个角色,管理员,普通用户,游客,而这三个角色的基本信息是相同的,所以DB中相应的有两个表:

        1. User表,存放用户的基本信息,假设有以下字段:

            uid userName password lid

            注:lid在逻辑上是后面我要说的Limits表的外键。

       2. Limits表,存放用户的权限信息,有以下字段:

            lid limits

            limits是一个int类型的值,假设1代表普通用户,2代表管理员,由于游客在User表中没有记录,所以就不记在这个表中。

      那么对应的模型有:

      表模型有两个:User和Limits。

      现在我们以User这个模型来举例,这个类的基本信息如下:

1
2 class User {
3     private $_pk = 'uid';
4     private $_tableInfo = array(
5         'uid','userName','password','lid'
6     );
7     private $_tablePrefix = '';
8     private $_tableName = '';
9 }

 

       我们用$_pk这个变量代表这个表的主键,将表的信息存储在$_tableInfo中,$_tablePrefix代表表前缀,它可以防止多个应用的表重名,$_tableName,逻辑上这个表的名字,物理上表的名字为$tablePrefix . $_tablePrefix,当然,你也可以定义$_tableSuffix。

        对于关系模型来说,User表和Limits表之间的关系就是lid,所以,这个关系模型就可以写成这样:

01
02 class Lid {
03     private $_tables = array(
04         'User','Limits'
05     );
06     private $_tablesInfo = array(
07         'Users' => array(
08             'uid','userName','password','lid'
09         ),
10         'Limits' => array(
11             'lid','limits'
12         )
13     );
14         private $_mainTable = 'User';
15 }

         这里的$_tables指定到底是哪两个表之间的关联,$_tablesInfo还是指明表的基本信息,$_mainTable指明到底那一个表是主表。

         如果现在模型这一层就抽象成这样,那么假设在控制器中就直接实例化一个表模型或者关系模型,还是举一个例子吧!

         假设在User模型中有一个 方法login来完成登录,返回值为bool,这个方法的具体实现我们不管,那么在控制器中的代码如下:

01
02 class IndexController extends Controller {
03     public function index() {
04         $userName = empty($_GET['userName']) ? null : trim($_GET['userName']);
05         $password = empty($_GET['password']) ? null : trim($_GET['password']);
06         if((null === $userName) || (null === $password)) {
07             //跳转
08         } else {
09             $user = new User();
10             if($user->login($userName,$password)) {
11                 //Success,然后设置session或COOKIE,跳转
12             } else {
13                 //Fail,跳转
14             }
15         }
16     }
17 }

         这样做有什么问题吗?

          1. 对于表模型来说,login这个单词不太恰当,login这个词与业务关联太大,对于一个简单的表模型来说,最好能够很好的和业务隔离,它只完成一些基本操作;

          2. 如果现在不是直接访问DB,而是首先在memcache中查看是否有这个key,如果有,直接从memcache中去,如果没有,访问DB,然后将值设置到memcache中,然后返回,这种情况,就只有在控制器中写很多本来不属于控制器该写的代码了;

          所以,这儿,我们再抽象出来一层,这一层处于控制器和表模型与关系模型之间,这样,控制器直接访问的是这一层,这一层跟业务紧密结合,就可以让表模型和关系模型从复杂的业务中脱离出来,这样以后对于项目的扩展很有帮助。

           还是使用刚才这个例子,假设这个类的名字为UserModel,那么它的代码如下:

01
02 class UserModel {
03     public function login($userName,$password) {
04         $user = new User();
05         if($user->isPasswordCorrect($userName,$password)) {
06             //设置COOKIE或session,不写了
07             return true;
08         } else {
09             //密码不正确
10             return false;
11         }
12     }
13 }

 

           这样控制器这一块儿的代码就变成这样了:

01
02 class IndexController extends Controller {
03     public function index() {
04         $userName = empty($_GET['userName']) ? null : trim($_GET['userName']);
05         $password = empty($_GET['password']) ? null : trim($_GET['password']);
06         if((null === $userName) || (null === $password)) {
07             //跳转
08         } else {
09             $userModel = new UserModel();
10             if($userModel->login($userName,$password)) {
11                 //Success,跳转
12             } else {
13                 //Fail,跳转
14             }
15         }
16     }
17 }

            控制器中没有业务逻辑的处理代码了,User这个类的命名也变得很有意义了,isPasswordCorrect。

            我们将处于控制器和表模型与关系模型之间的这一层称为模型(和Java的service这一层很类似),这样,对于编写控制器的程序员来说,它只需要查看模型这一层的代码就可以了,到底模型这一层调用的是关系模型或者表模型都没有关系。

            实际上我们现在将模型完全看成了对DB的操作了,实际上模型不一定是操作DB的,现在我们还可以考虑一下,在模型这一层将Cache抽象出来,现在编写一个cache的接口ICache:

1
2 interface ICache {
3     public function connect();
4     public function get($name);
5     public function set($name,$val,$expire = null);
6     public function have($name);
7     public function remove($name);
8     public function clear();
9 }

           然后针对不同的Cache,只要实现这个接口即可。

           刚才我们考虑的是Cache,实际上还有nosql,现在有很多nosql,我们可以将一些共有的操作抽象出来变成INosql,当然,这儿我就不写了。


推荐阅读
  • 在探讨如何高效处理大规模数据报表的分页展示之前,首先需要明确导致报表加载缓慢的主要原因。通常情况下,这主要是由于两个方面:一是查询条件过于宽泛,使得数据库返回的结果集包含数百万甚至更多的记录;二是前端渲染性能不足,无法高效处理大量数据。为了优化这一过程,可以从以下几个方面入手:优化查询条件,减少不必要的数据返回;采用分页查询技术,每次仅加载所需的数据;利用缓存机制,减少对数据库的频繁访问;提升前端渲染效率,使用虚拟滚动等技术提高用户体验。 ... [详细]
  • 在Linux环境下,本文详细探讨了Apache服务器中CGI技术的应用与实现。首先,通过使用yum包管理器安装了必要的软件,如PHP。安装完成后,对Apache服务器进行了配置,确保CGI功能正常运行。此外,还介绍了如何编写和调试CGI脚本,以及如何在实际环境中部署这些脚本以提供动态网页内容。实验结果表明,通过合理的配置和优化,Apache服务器能够高效地支持CGI应用程序,为用户提供丰富的交互体验。 ... [详细]
  • Issue with the Reserved Term HOSTS in System Configuration ... [详细]
  • 题目描述非常吸引人。每颗星星可以通过其在窗口的左下角和右上角位置构建两条扫描线,从而将问题转化为区间增减和求最大值的操作。需要注意的是,位于边界的星星不应计入结果,因此在处理时应分别对左右边界进行适当的增减调整。此外,利用线段树和离散化技术可以显著提高算法效率,确保在大规模数据下的性能表现。 ... [详细]
  • 本文探讨了如何在C#中实现USB条形码扫描仪的数据读取,并自动过滤掉键盘输入,即使不知道设备的供应商ID(VID)和产品ID(PID)。通过详细的技术指导和代码示例,展示了如何高效地处理条形码数据,确保系统能够准确识别并忽略来自键盘的干扰信号。该方法适用于多种USB条形码扫描仪,无需额外配置设备信息。 ... [详细]
  • 在《PHP应用性能优化实战指南:从理论到实践的全面解析》一文中,作者分享了一次实际的PHP应用优化经验。文章回顾了先前进行的一次优化项目,指出即使系统运行时间较长后出现的各种问题和性能瓶颈,通过采用一些通用的优化策略仍然能够有效解决。文中不仅详细阐述了优化的具体步骤和方法,还结合实例分析了优化前后的性能对比,为读者提供了宝贵的参考和借鉴。 ... [详细]
  • Vuex 实战进阶:构建高效笔记本应用(第二篇)
    在上一篇文章中,我们初步探讨了 Vuex 在该项目中的应用。本文将深入解析整个项目的架构设计。首先回顾 `main.js` 的内容,然后重点分析 `App.vue` 文件,其中引入了 `Toolbar.vue` 和 `NodeList.vue` 组件,详细说明它们在应用中的作用和交互方式。通过这些组件的协同工作,我们将展示如何构建一个高效且响应迅速的笔记本应用。 ... [详细]
  • Photoshop教程第五讲:使用套索工具精准抠图技巧
    在本节Photoshop教程中,我们将深入探讨如何利用套索工具实现精准的图像抠图。通过详细的操作步骤和实用技巧,帮助用户掌握套索工具的多种使用方法,提升图像处理的精细度和效率。 ... [详细]
  • 本文作为“实现简易版Spring系列”的第五篇,继前文深入探讨了Spring框架的核心技术之一——控制反转(IoC)之后,将重点转向另一个关键技术——面向切面编程(AOP)。对于使用Spring框架进行开发的开发者来说,AOP是一个不可或缺的概念。了解AOP的背景及其基本原理,对于掌握这一技术至关重要。本文将通过具体示例,详细解析AOP的实现机制,帮助读者更好地理解和应用这一技术。 ... [详细]
  • 本文详细解析了如何使用 jQuery 实现一个在浏览器地址栏运行的射击游戏。通过源代码分析,展示了关键的 JavaScript 技术和实现方法,并提供了在线演示链接供读者参考。此外,还介绍了如何在 Visual Studio Code 中进行开发和调试,为开发者提供了实用的技巧和建议。 ... [详细]
  • 本文首先对信息漏洞的基础知识进行了概述,重点介绍了几种常见的信息泄露途径。具体包括目录遍历、PHPINFO信息泄露以及备份文件的不当下载。其中,备份文件下载涉及网站源代码、`.bak`文件、Vim缓存文件和`DS_Store`文件等。目录遍历漏洞的详细分析为后续深入研究奠定了基础。 ... [详细]
  • React组件是构成用户界面的基本单元,每个组件都封装了特定的功能和逻辑,具备高度的独立性和可复用性。通过将不同大小和功能的组件组合在一起,可以构建出复杂且功能丰富的页面,类似于拼图游戏中的各个部分,最终形成一个完整的视觉效果。 ... [详细]
  • 在面对不确定性的挑战时,卓越的操作者通常会采用七大策略来有效管理和减轻风险,这些策略同样适用于职业发展和个人生活。具体而言,这七大风险管理策略包括:1. 克服恐惧心理卓越的操作者能够正视并克服内心的恐惧,保持冷静和理性,从而做出更加明智的决策。这一能力不仅有助于在市场波动中保持稳定,也能在职业生涯和个人生活中发挥重要作用。 ... [详细]
  • 《题画山水屏风》译文与原文赏析:唐代诗人张九龄的艺术解读 ... [详细]
  • 《已亥杂诗 第139首》译文与原文鉴赏:清代文学家龚自珍的诗歌艺术探析 ... [详细]
author-avatar
励志天涯网
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有