Home > Article > Backend Development > PHP学习笔记,自己动手写个MVC的框架
最新在大家自己的博客的过程中,发现各种开源的博客系统都或多或少的用起来别扭.于是想动手自己写个博客系统.既然写,就想好好写.那就先写个MVC框架.一点一点来.写的过程中有很多想法.还希望大家能够多多指正.我在这里先把它们记录下来.下面是我对这个系统的一些想法.
1,我要实现一个模块话的博客系统,能够进行二次开发.
2,我要实现apache和nginx的rewrite功能.
3,我要实现对多数据库的支持.包括mongodb和mysql 还有mysqli.等.
4,我要把smarty用起来.
好,下面动手开始写.首先一点我得设计一下这个系统.大概的目录结构是下面这个样子的(我参考了PHPCMS).
/
./log--这个是日志目录.我想把日志记录在这个里面
./system ---这个是系统目录
./lib--这个是系统库.关键的东西都放在这里面
./classes -- 这个是系统的相关类
./configs--这个是系统的配置文件目录
./model--这个是各个数据模型的目录
./modules--这个是各个模块的目录
./base.php--所有的请求都路由到这个文件上面来了,再有这个文件来分发.
./index --我把index也当成一个模块来写
./templates--这个是模板目录
./default--这个是默认的模板目录
./cache--这个是cache目录.
index.php-----------这个是单入口文件,用来路由相关的请求.
MVC框架要把model和ctrl还有view分开.那么就需要url的路由.至于伪静态什么的,后面再说.要先满足最基本的需求.
MVC框架要有个单入口文件,于是第一个文件产生了.就是根目录下面的index.php,这个文件用来接收所有的请求,也就是说所有的请求都是从这个入口进来的.关于单入口的好处,请自行搜索脑补.
每一个数据模型,按照我的理解,应该对应一张或者多张表.比如文章模型.可以单独对文章表.也可以对应作者表and文章表and评论表.
涉及到数据模型就要与数据库打交道了.先不管跟数据库打交道.我最先要实现的是能够路由我的URL
最简单的url http://域名.com要路由到index模块下的index.php文件里面的index控制器并且执行这个控制器的默认方法(我设置成了init);
既然要有默认的路由参数,我就建立了一个文件.叫default_arg.config.php 存放在/system/lib/configs/default_arg.config.php里面用来返回默认的参数
里面的内容大概是介个样子滴:
<?php return array( 'default' => array( 'm'=>'index', 'c'=>'index', 'a'=>'init', ), );
当我想引用这些配置的时候 .我只需要如下的调用
$configs = include_once($file);
就能够将这个大数组赋值到configs上面,如果没有参数,就使用默认的参数.把默认参数拼接到URL上面.我的默认首页就变成了
http://域名.com/index.php?m=index&c=index&a=init
这一切的功能是怎么实现的哦?
既然访问的是index.php,那就从index.php开始看,其实index里面就几行,
define('ROOT_PATH',dirname(__FILE__));//定义一个系统路径require_once(ROOT_PATH.DIRECTORY_SEPARATOR.'system'.DIRECTORY_SEPARATOR.'base.php');//引用框架的基础类$sys = base::getInstance();//得到基础类的实例,基础类是一个单例类$sys->init();//调用单例类的init方法
其实DIRECTORY_SEPARATOR就是个/,我们可以这样理解.引用了框架里面的base文件.然后调用了里面的getInstance方法.得到了一个实力,最后调用了init方法.
为什么要使用单例类,可以自行百度.这里用单例类比较科学.后面我会把整个类贴上来,下面用到什么就贴什么.
再看看base类里面的getInstance干了些什么.
class base{ public static $sys; private function __construct(){ return false; } public static function getInstance(){ if(!(self::$sys instanceof self)){ self::$sys = new self(); } return self::$sys; }
因为是单例类,我把base里面的构造方法声明成了私有的.这是为了防止被new关键字从外部new这个类.为了保证所有操作都是由单一实例来完成的.这个类是不允许在外部new的.
在看看getInstance方法.先判断自己的$sys变量是不是自己的实例.如果不是就将自己的实例赋值给$sys,如果是则不做操作,最后 返回了这个类自己的一个实例.
在看看init方法做了些什么
public static function init(){ self::sys_class('model'); self::sys_class('ctrl'); $args = self::__explan_arg(); $ctrl = self::__load_ctrl($args['m'],$args['c']); call_user_func(array($ctrl,$args['a'])); }
我在这个基础类里面写了几个方法.如果方法名称前面有两个下划线,就是私有的方法. 有个sys_class就是在指定目录加载系统了.这个目录是/system/lib/class/. 这里加载了model类和ctrl类,就是模型类的基类和控制器类的基类.这个ctrl类是所有控制器的基类.里面可以写一些公共的方法.比如说在构造方法让类中有一个base的实例神马的.让所有的控制器都集成自这个类.这个model类目前还没用到.但是以后的数据模型都应该是来自这个model类的.后面会说.
然后我调用了__explan_arg方法.这个方法就是来解析get得到的参数的.
private static function __explan_arg(){ $default_arg = self::sys_config('default_arg'); $args['m'] = isset($_GET['m'])?$_GET['m']:$default_arg['m']; $args['c'] = isset($_GET['c'])?$_GET['c']:$default_arg['c']; $args['a'] = isset($_GET['a'])?$_GET['a']:$default_arg['a']; return $args; }
我在第一行使用了一个sys_config方法来加载默认参数.这个方法就是在系统的/system/lib/configs/目录下面找到对应的配置文件,上面已经说过了怎么把数组返回.这样当GET里面没有相应的参数的时候就会使用默认的参数.接下来我们调用了__load_ctrl方法加载了相应的控制器.传入了m和c.这个方法实现的就是到m所指定的目录下面找到c这个文件并且实例化一个c这个类(也就是相应的控制器类.)并且返回相应控制器类的实例.然后我调用了一个call_user_func方法.因为我们没办法在程序里像下面的样子来调用控制器的方法
$ctrl->$args['a'];//这样是没办法调用的
所以我们使用了call_user_func方法来调用相应控制器的方法.
好了,现在再缕缕我们程序的流程.首先访问了index.php-->index.php定义了一个路径,去引用了base类.并且得到了一个base类的实例.还调用了base的init方法.-->base的init方法做了下面的事情-->先去引用了基类model和ctrl-->去解析了url中的参数,得到了m,c,a-->通过m,c来引用相应的控制器,-->调用相应控制器的a方法.然后就会得到相应的输出了.
到此为止我们的框架控制器部分基本算是完成了,默认的args是index,index,init.我们在/modules/index/里面建立一个index.php文件.里面写如下的内容.
<?phpclass c_index extends ctrl{ public function __construct(){ parent::__construct(); } public function init(){ echo "hello my mvc!"; }}
再来访问我们的根域名,那么我们就会得到hello my mvc.这句话的输出.
----------------------------------------------------------------------------------------------------------------------------------
毕竟是自己个人的思路.如果有什么不妥的地方,欢迎大家拍砖,也希望大家能够一起来参与讨论,最近看到php的相关板块不像以前那么火了.还希望大家能够多多来参与发帖和讨论.