0

0

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

php中文网

php中文网

发布时间:2016-06-21 08:50:39

|

1377人浏览过

|

来源于php中文网

原创

 

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

        如果是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,当然,这儿我就不写了。



PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

1044

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

334

2026.02.13

Flutter跨平台开发与状态管理实战
Flutter跨平台开发与状态管理实战

本专题围绕Flutter框架展开,系统讲解跨平台UI构建原理与状态管理方案。内容涵盖Widget生命周期、路由管理、Provider与Bloc状态管理模式、网络请求封装及性能优化技巧。通过实战项目演示,帮助开发者构建流畅、可维护的跨平台移动应用。

213

2026.02.13

TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

35

2026.02.13

Redis高可用架构与分布式缓存实战
Redis高可用架构与分布式缓存实战

本专题围绕 Redis 在高并发系统中的应用展开,系统讲解主从复制、哨兵机制、Cluster 集群模式及数据分片原理。内容涵盖缓存穿透与雪崩解决方案、分布式锁实现、热点数据优化及持久化策略。通过真实业务场景演示,帮助开发者构建高可用、可扩展的分布式缓存系统。

111

2026.02.13

c语言 数据类型
c语言 数据类型

本专题整合了c语言数据类型相关内容,阅读专题下面的文章了解更多详细内容。

77

2026.02.12

雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法
雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法

本专题系统整理雨课堂网页版官方入口及在线登录方式,涵盖账号登录流程、官方直连入口及平台访问方法说明,帮助师生用户快速进入雨课堂在线教学平台,实现便捷、高效的课程学习与教学管理体验。

17

2026.02.12

豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法
豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法

本专题汇总豆包AI官方网页版入口及在线使用方式,涵盖智能写作工具、图片生成体验入口和官网登录方法,帮助用户快速直达豆包AI平台,高效完成文本创作与AI生图任务,实现便捷智能创作体验。

813

2026.02.12

PostgreSQL性能优化与索引调优实战
PostgreSQL性能优化与索引调优实战

本专题面向后端开发与数据库工程师,深入讲解 PostgreSQL 查询优化原理与索引机制。内容包括执行计划分析、常见索引类型对比、慢查询优化策略、事务隔离级别以及高并发场景下的性能调优技巧。通过实战案例解析,帮助开发者提升数据库响应速度与系统稳定性。

97

2026.02.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号