PHP实现MVC开发:一个简单的MVC
今天研究了下PHP MVC结构,所以决定自己写个简单的MVC,以待以后有空再丰富。
至于什么MVC结构,其实就是三个Model,Contraller,View单词的简称,Model,主要任务就是把数据库或者其他文件系统的数据按 照我们需要的方式读取出来,View,主要负责页面的,把数据以html的形式显示给用户,Controller,主要负责业务逻辑,根据用户的 Request进行请求的分配,比如说显示登陆界面,就需要调用一个控制器userController的方法loginAction来显示.
下面我们用PHP来创建一个简单的MVC结构系统。
首先创建单点入口,即bootstrap文件index.php,作为整个MVC系统的唯一入口。什么是单点入口呢?所谓单点入口就是整个应用程序只有一 个入口,所有的实现都通过这个入口来转发。为什么要做到单点入口呢?单点入口有几大好处:第一、一些系统全局处理的变量,类,方法都可以在这里进行处理。 比如说你要对数据进行初步的过滤,你要模拟session处理,你要定义一些全局变量,甚至你要注册一些对象或者变量到注册器里面。第二、程序的架构更加 清晰明了。当然好处还有很多的。:)
<?php include ( "core/ini.php" ); initializer::initialize(); $router = loader::load( "router" ); dispatcher::dispatch( $router ); ?>这个文件就只有4句,我们现在一句句来分析。
include(]core/ini.php]);
我们来看core/ini.php
<?php set_include_path(get_include_path() . PATH_SEPARATOR . "core/main" ); //set_include_path — Sets the include_path configuration option function __autoload( $object ){ require_once ( "{$object}.php" ); } ?>这个文件首先设置了include_path,也就是我们如果要找包含的文件,告诉系统在这个目录下查找。其实我们定义__autoload()方法,这个方法是在PHP5增加的,就是当我们实例化一个函数的时候,如果本文件没有,就会自动去加载文件。官方的解释是:
Many developers writing object-oriented applications create one PHP source file per-class definition. One of the biggest annoyances is having to write a long list of needed includes at the beginning of each script (one for each class).
In PHP 5, this is no longer necessary. You may define an __autoload function which is automatically called in case you are trying to use a class/interface which hasn’t been defined yet. By calling this function the scripting engine is given a last chance to load the class before PHP fails with an error.
接下来我们看下面一句:
initializer::initialize();
这就话就是调用initializer类的一个静态函数initialize,因为我们在ini.php,设置了include_path,以及定义了__autoload,所以程序会自动在core/main目录查找initializer.php.
initializer.php文件如下:
<?php class initializer { public static function initialize() { set_include_path(get_include_path().PATH_SEPARATOR . "core/main" ); set_include_path(get_include_path().PATH_SEPARATOR . "core/main/cache" ); set_include_path(get_include_path().PATH_SEPARATOR . "core/helpers" ); set_include_path(get_include_path().PATH_SEPARATOR . "core/libraries" ); set_include_path(get_include_path().PATH_SEPARATOR . "app/controllers" ); set_include_path(get_include_path().PATH_SEPARATOR. "app/models" ); set_include_path(get_include_path().PATH_SEPARATOR. "app/views" ); //include_once("core/config/config.php"); } } ?>这个函数很简单,就只定义了一个静态函数,initialize函数,这个函数就是设置include_path,这样,以后如果包含文件,或者__autoload,就会去这些目录下查找。
OK,我们继续,看第三句:
$router = loader::load(]router]);
这句话也很简单,就是加载loader函数的静态函数load,下面我们来loader.php:
<?php class loader { private static $loaded = array (); public static function load( $object ){ $valid = array ( "library" , "view" , "model" , "helper" , "router" , "config" , "hook" , "cache" , "db" ); if (!in_array( $object , $valid )){ throw new Exception( "Not a valid object '{$object}' to load" ); } if ( empty empty (self:: $loaded [ $object ])){ self:: $loaded [ $object ]= new $object (); } return self:: $loaded [ $object ]; } } ?>这个文件就是去加载对象,因为以后我们可能会丰富这个MVC系统,会有model,helper,config等等的组件,如果加载的组件不在有效 的范围内,我们抛出一个异常,如果在的话,我们实例化一个对象,其实这里用了单件设计模式,也就是这个对象其实就只能是一个实例化对象,如果没有实例化,创建一个,如果存在的,则不实例化.
好,因为我们现在要加载的是router组件,所以我们看下router.php文件,这个文件的作用就是映射URL,对URL进行解析.
router.php:
<?php class router { private $route ; private $controller ; private $action ; private $params ; public function __construct() { $path = array_keys ( $_GET ); if (!isset( $path [0])){ if (! empty empty ( $default_controller )) $path [0] = $default_controller ; else $path [0] = "index" ; } $route = $path [0]; $this ->route = $route ; $routeParts = split( "/" , $route ); $this ->controller= $routeParts [0]; $this ->action=isset( $routeParts [1])? $routeParts [1]: "base" ; array_shift ( $routeParts ); array_shift ( $routeParts ); $this ->params= $routeParts ; } public function getAction() { if ( empty empty ( $this ->action)) $this ->action= "main" ; return $this ->action; } public function getController() { return $this ->controller; } public function getParams() { return $this ->params; } } ?>我们可以看到,首先我们是拿到$_GET,用户Request的URL,然后从URL里我们解析出Controller和Action,以及Params
OK我们看最后一句,就是
dispatcher::dispatch($router);
这句话的意思很明了,就是拿到URL解析的结果,然后通过dispatcher来分发controlloer及action来Response给用户
好,我们来看下dispatcher.php文件:
<? class dispatcher { public static function dispatch( $router ) { global $app ; ob_start(); $start = microtime(true); $controller = $router ->getController(); $action = $router ->getAction(); $params = $router ->getParams(); $controllerfile = "app/controllers/{$controller}.php" ; if ( file_exists ( $controllerfile )){ require_once ( $controllerfile ); $app = new $controller (); $app ->setParams( $params ); $app -> $action (); if (isset( $start )) echo " Tota1l time for dispatching is : ".(microtime(true)-$start)." seconds. "; $output = ob_get_clean(); echo $output ; } else { throw new Exception( "Controller not found" ); } } } ?>这个类很明显,就是拿到$router来,寻找文件中的controller和action来回应用户的请求.
OK,我们一个简单的,MVC结构,就这样,当然这里还不能算是一个很完整的MVC,因为这里还没有涉及到View和Model,有空我再这里丰富.
我们来写个Controller文件来测试下上面的这个系统,我们在app/controllers/下创建一个user.php文件,user.php:
<?php class user { function base() { } public function login() { echo 'login html page' ; } public function register() { echo 'register html page' ; } public function setParams( $params ){ var_dump( $params ); } } ?>查看更多关于PHP实现MVC开发:一个简单的MVC - Thinkphp的详细内容...