好得很程序员自学网

<tfoot draggable='sEl'></tfoot>

PHP实现MVC开发:一个简单的MVC - Thinkphp

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的详细内容...

  阅读:88次