«
理解PHP中的MVC编程之控制器

时间:2008-5-31    作者:Deri    分类: 分享


   <p>  简单来讲,控制器的作用就是接受请求。它使用获取的方法,在这里是通过URI,载入一个功能模块来刷新或者提交一个表述层。控制器将使用$_GET自动全局变量来判断载入哪一个模块。<p>  一个请求的例子,看起来像这样:</p><p>  http://example.com/index.php?module=login</p><p>  这看起来很简单,但是在实现的过程中却不是。这里是几个控制器能识别的argument部分:</p><p>  module定义了使用哪一个模块,如users模块</p><p>  class定义了使用哪一个功能类,如你想让用户login还是logout</p><p>  event定义了使用哪一个具体事件</p><p>  这样一个更复杂的例子可以解释上面的各个argument最终组成的请求URL:</p><p>  http://example.com/index.php?module=users&class=login</p><p>  这段请求告诉控制器应该载入users模块,然后是login类,最后因为没有定义具体事件,所以运行login::__default()默认事件。</p><p>  以下是具体代码部分:</p><code><?php<br />  /**<br />   * index.php<br />   *<br />   * @author Joe Stump <joe@joestump.net><br />   * @copyright Joe Stump <joe@joestump.net><br />   * @license http://www.opensource.org/licenses/gpl-license.php<br />   * @package Framework<br />  */<br /> require_once('config.php');<br /> // {{{ __autoload($class)<br />  /**<br />   * __autoload<br />   *<br />   * Autoload is ran by PHP 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 <joe@joestump.net><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 /> if (isset($_GET['module'])) {<br />   $module = $_GET['module'];<br />   if (isset($_GET['event'])) {<br />    $event = $_GET['event'];<br />   } else {<br />    $event = '__default';<br />   }<br /> if (isset($_GET['class'])) {<br />   $class = $_GET['class'];<br />  } else {<br />   $class = $module;<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 />    $instance->moduleName = $module;<br />     if ($instance->authenticate()) {<br />      try {<br />       $result = $instance->$event();<br />       if (!PEAR::isError($result)) {<br />        $presenter = FR_Presenter::factory($instance->presenter,$instance);<br />      if (!PEAR::isError($presenter)) {<br />        $presenter->display();<br />       } else {<br />        die($presenter->getMessage());<br />       }<br />      }<br />     } catch (Exception $error) {<br />      die($error->getMessage());<br />     }<br />    } else {<br />     die("You do not have access to the requested page!");<br />    }<br />   } catch (Exception $error) {<br />    die($error->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 />?></code></p>
<p> </p>

   <p>  接下来是以上代码具体的注释:</p><p>  载入“config.php”</p><p>  定义__autoload()函数。这是PHP5里面的一个新函数,方便动态地载入各个类。</p><p>  如果一个argument被定义,那么载入相关的模块、类和具体事件</p><p>  接下来就是一些判断以及错误的具体操作</p><p>  最后一切无误后就载入表述层</p><p>  【友好URL】</p><p>  如果你觉得上面例子讲到的请求URL让你觉得不舒服的话,那么就用mod_rewrite来实现友好URL吧。接下来是作者给这个框架写的实际重写标准代码:</p><code>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 />Extending the Controller</code><p>  【扩展控制器】</p></p><p>  拥有一个集中控制器的一点好处就是你加入一些功能后,马上就能通过控制器体现出来。以下是几个可以扩展一下这个控制器的点子,使这个框架的整体能力更加强大:</p><p>  你可以使用PHP5里一个新东西SoapServer来自动检测一个请求是否为SOAP</p><p>  你可以使用控制器来过滤所有的自动全局变量如$_GET和$_POST以防止恶意HTML代码等</p><p>  你可以使用控制器即时地转换表述层,比如从默认的方式转到PDF方式</p><p>  你可以直接在控制器中加入缓存机制,这样的好处是应用程序整体都能使用到缓存以提高效率</p><p>  当然,需要注意一点的是,你在控制器中所增加的功能将体现在程序全局。如你想过滤所有的自动全局变量,但是很多应用程序的管理员需要使用到一些HTML代码,反而成为一件棘手的事情(译者注:本人的想法是可以加一个if条件语句,在加载特定模块时不应用过滤功能即可)。</p></p>