发新话题
打印

[php]贯彻PHP中的MVC之控制器篇

[php]贯彻PHP中的MVC之控制器篇

The Controller 【控制器】 <BR><BR>简单来讲,控制器的作用就是接受请求。它使用获取的方法,在这里是通过URI,载入一个功能模块来刷新或者提交一个表述层。控制器将使用$_GET自动全局变量来判断载入哪一个模块。 <BR>一个请求的例子,看起来像这样: <BR>http://example.com/index.php?module=login <BR>这看起来很简单,但是在实现的过程中却不是。这里是几个控制器能识别的argument部分: <BR>module定义了使用哪一个模块,如users模块  <BR>class定义了使用哪一个功能类,如你想让用户login还是logout  <BR>event定义了使用哪一个具体事件  <BR><BR>这样一个更复杂的例子可以解释上面的各个argument最终组成的请求URL: <BR>http://example.com/index.php?module=users&class=login <BR>这段请求告诉控制器应该载入users模块,然后是login类,最后因为没有定义具体事件,所以运行login::__default()默认事件。 <BR>以下是具体代码部分: <BR>&lt;?php  <BR><BR>  /** <BR>  * index.php <BR>  * <BR>  * @author Joe Stump &lt;joe@joestump.net&gt; <BR>  * @copyright Joe Stump &lt;joe@joestump.net&gt; <BR>  * @license http://www.opensource.org/licenses/gpl-license.php <BR>  * @package Framework <BR>  */  <BR><BR>  require_once(’config.php’);  <BR><BR>  // {{{ __autoload($class) <BR>  /** <BR>  * __autoload <BR>  * <BR>  * Autoload is ran by HP when it can’t find a class it is trying to load. <BR>  * By naming our classes intelligently we should be able to load most classes <BR>  * dynamically. <BR>  * <BR>  * @author Joe Stump &lt;joe@joestump.net&gt; <BR>  * @param string $class Class name we’re trying to load <BR>  * @return void <BR>  * @package Framework <BR>  */ <BR>  function __autoload($class) <BR>  { <BR>      $file = str_replace(’_’,’/’,substr($class,2)).’.php’;      <BR>      require_once(FR_BASE_PATH.’/includes/’.$file); <BR>  } <BR>  // }}}  <BR><BR>  if (isset($_GET[’module’])) { <BR>      $module = $_GET[’module’]; <BR>      if (isset($_GET[’event’])) { <BR>          $event = $_GET[’event’]; <BR>      } else { <BR>          $event = ’__default’; <BR>      }  <BR><BR>      if (isset($_GET[’class’])) { <BR>          $class = $_GET[’class’]; <BR>      } else { <BR>          $class = $module; <BR>      }  <BR><BR>      $classFile = FR_BASE_PATH.’/modules/’.$module.’/’.$class.’.php’; <BR>      if (file_exists($classFile)) { <BR>          require_once($classFile); <BR>          if (class_exists($class)) { <BR>              try { <BR>                  $instance = new $class(); <BR>                  if (!FR_Module::isValid($instance)) { <BR>                      die("Requested module is not a valid framework module!"; <BR>                  }  <BR><BR>                  $instance-&gt;moduleName = $module; <BR>                  if ($instance-&gt;authenticate()) { <BR>                      try { <BR>                         $result = $instance-&gt;$event(); <BR>                          if (!PEAR::isError($result)) { <BR>                              $presenter = FR_Presenter::factory( <BR>                                  $instance-&gt;presenter,$instance <BR>                              ;  <BR><BR>                              if (!PEAR::isError($presenter)) { <BR>                                  $presenter-&gt;display(); <BR>                              } else { <BR>                                  die($presenter-&gt;getMessage()); <BR>                              } <BR>                         } <BR>                      } catch (Exception $error) { <BR>                          die($error-&gt;getMessage()); <BR>                      } <BR>                  } else { <BR>                      die("You do not have access to the requested page!"; <BR>                  } <BR>              } catch (Exception $error) { <BR>                  die($error-&gt;getMessage());    <BR>              } <BR>          } else { <BR>              die("An valid module for your request was not found");         <BR>          } <BR>      } else { <BR>          die("Could not find: $classFile");         <BR>      } <BR>  } else { <BR>      die("A valid module was not specified"); <BR>  }  <BR><BR>?&gt;  <BR><BR>接下来是以上代码具体的注释: <BR>载入“config.php”  <BR>定义__autoload()函数。这是PHP5里面的一个新函数,方便动态地载入各个类。  <BR>如果一个argument被定义,那么载入相关的模块、类和具体事件  <BR>接下来就是一些判断以及错误的具体操作  <BR>最后一切无误后就载入表述层  <BR>  <BR>retty URLs 【友好URL】 <BR><BR>如果你觉得上面例子讲到的请求URL让你觉得不舒服的话,那么就用mod_rewrite来实现友好URL吧。接下来是作者给这个框架写的实际重写标准代码: <BR>RewriteEngine On <BR># Change the URI here to whatever you want your homepage to be <BR>RewriteRule ^/$ /index.php?module=welcome [L,QSA] <BR># Changes /index.php?module=welcome to /welcome <BR>RewriteCond %{DOCUMENT_ROOT}/%{REQUEST_FILENAME} !-d <BR>RewriteCond %{DOCUMENT_ROOT}/%{REQUEST_FILENAME} !-f <BR>RewriteRule ^/([^/]*)$ /index.php?module=$1 [L,QSA] <BR># Changes /index.php?module=users&class=login to /users/login <BR>RewriteCond %{DOCUMENT_ROOT}/%{REQUEST_FILENAME} !-d <BR>RewriteCond %{DOCUMENT_ROOT}/%{REQUEST_FILENAME} !-f <BR>RewriteRule ^/([^/]*)/([^/]*)$ /index.php?module=$1&class=$2 [L,QSA] <BR># Changes /index.php?module=users&class=login&event=foo <BR>#     to /users/login/foo.html <BR>RewriteCond %{DOCUMENT_ROOT}/%{REQUEST_FILENAME} !-d <BR>RewriteCond %{DOCUMENT_ROOT}/%{REQUEST_FILENAME} !-f <BR>RewriteRule ^/([^/]*)/([^/]*)/([^/]*).html$ \ <BR>/index.php?module=$1&class=$2&event=$3 [L,QSA]  <BR><BR>Extending the Controller 【扩展控制器】 <BR><BR>拥有一个集中控制器的一点好处就是你加入一些功能后,马上就能通过控制器体现出来。以下是几个可以扩展一下这个控制器的点子,使这个框架的整体能力更加强大: <BR>你可以使用PHP5里一个新东西SoapServer来自动检测一个请求是否为SOAP  <BR><BR>你可以使用控制器来过滤所有的自动全局变量如$_GET和$_POST以防止恶意HTML代码等  <BR><BR>你可以使用控制器即时地转换表述层,比如从默认的方式转到PDF方式 <BR><BR>你可以直接在控制器中加入缓存机制,这样的好处是应用程序整体都能使用到缓存以提高效率  <BR>  <BR>当然,需要注意一点的是,你在控制器中所增加的功能将体现在程序全局。如你想过滤所有的自动全局变量,但是很多应用程序的管理员需要使用到一些HTML代码,反而成为一件棘手的事情(译者注:本人的想法是可以加一个if条件语句,在加载特定模块时不应用过滤功能即可)。

TOP

发新话题