2.3 前端控制器

翻译 2015年07月06日 22:40:46

2.3.1. 概述
Zend_Controller_Front实现了模型-视图-控制器 (MVC)应用程序的前端控制器模式。目的在于初始化请求环境,并路由到来的请求,接着分发任何发现的动作;收集所有的响应,在整个过程完成时就其返回。

Zend_Controller_Front也实现了单件(Singleton)模式,意味着任何时候,都只可能有一个有效实例。这使得它可以作为注册表,供分发过程中的其他对象引用。

Zend_Controller_Front自己注册了一个插件经纪人(plugin broker),允许插件观测它所触发的各种事件。大多数情况下,这将使得开发人员有机会裁剪站点的分发过程,而无需通过扩展前端控制器增加功能。

前端控制器最至少需要一个或多个包含动作控制器的目录的路径来完成工作。还有大量的方法可供调用,进一步裁剪前端控制器以及它的助手类环境。
这里写图片描述

2.3.2. 主要方法
前端控制器有很多建立其环境的访问器。但是,有三个是开启前端控制器功能的主要方法:

2.3.2.1. getInstance()
getInstance()方法用来获取前端控制器实例。因为前端控制器实现了单件模式,这可能是唯一创建前端控制器对象的方法。

<?php
$front = Zend_Controller_Front::getInstance();

2.3.2.2. setControllerDirectory() 和 addControllerDirectory()
setControllerDirectory()通知分发器到哪查找动作控制器action controller类文件。参数接受单一路径和模块/路径对关联数组。
例如:

// Set the default controller directory:
$front->setControllerDirectory('../application/controllers');

// Set several module directories at once:
$front->setControllerDirectory(array(
    'default' => '../application/controllers',
    'blog'    => '../modules/blog/controllers',
    'news'    => '../modules/news/controllers',
));

// Add a 'foo' module directory:
$front->addControllerDirectory('../modules/foo/controllers', 'foo');
注意 
如果使用addControllerDirectory()时不带模块名,将会为default模块设定目录——如果目录已设定,就覆盖掉。

可以通过getControllerDirectory()获取控制器目录的当前设置;它将返回一个模块/目录对关联数组。

2.3.2.3. dispatch()
dispatch(Zend_Controller_Request_Abstract $request = null, Zend_Controller_Response_Abstract $response = null)完成前端控制器最繁重的工作。该方法带有可选的参数请求对象和/或响应对象,允许开发人员为每一个传入定制的对象。

如果没有请求或者响应对象传入,dispatch()将检查先前注册的对象并使用,如果没有发现则创建默认的对象版本(它们两个都默认使用HTTP对象)。

类似的,dispatch()先检查已注册的路由器(router)和分发器(dispatcher)对象,如果没有发现则实例化它们的默认版本。

分发过程有三个不同的事件: 

路由(Routing)

分发(Dispatching)

响应(Response

路由只发生一次,当调用dispatch()时利用请求对象中的值。分发发生在一个循环中;请求可能指示 分发多个动作,或者控制器或插件可能重置请求对象,强制分发附加的动作。所有都完成后,前端控制器返回响应对象。

2.3.2.4. run()
Zend_Controller_Front::run($path)是静态方法,只带一个参数,就是指向包含控制器的目录的路径。它首先通过getInstance()获取前端控制器实例,然后通过setControllerDirectory()注册传入的路径,最后分发。

基本上,如果不要求定制前端控制器环境,run()是一个很方便的建立前端控制器环境的方法

<?php
// Instantiate front controller, set controller directory, and dispatch in one
// easy step:
Zend_Controller_Front::run('../application/controllers');

2.3.3. 环境访问器方法
除了上面所列的方法以外,还有很多访问器方法可以影响前端控制器环境 —— 因而也影响前端控制器代理(delegate)的类的环境。

resetInstance()方法清除当前的所有设置。主要用来测试,不过,
在希望将几个前端控制器连锁的地方也是很有用的
(but it can also be used for instances where you wish to chain 
together multiple front controllers)。 

(set|get)DefaultControllerName()
方法可以为默认的控制器指定另外一个名字
(否则使用'index'),以及获取当前值。
它们将代理分发器。 

(set|get)DefaultAction()方法可以为
默认的动作指定另外一个名字(否则使用'index')
,以及获取当前值。它们将代理分发器。 

(set|get)Request()方法指定分发过程中
使用的请求类或对象,以及获取当前的请求对象。
设置请求对象时,可以传入一个请求类的名字,
该方法将加载类文件并创建实例。 

(set|get)Router()方法指定分发过程中使用
的路由器类或对象,以及获取当前对象。设置路由器时,
可以传入一个路由器类的名字,该方法将加载类文件并创建实例。 

获取路由器对象的时候,首先检查是否已有一个,
如果没有,创建默认的路由器实例(rewrite路由器)。 

(set|get)BaseUrl()方法指定路由
请求时剥离(strip)的基地址(base URL),
以及获取当前值。这个值将在路由前提供给路由器。 

(set|get)Dispatcher()方法指定分发过程
中使用的分发器类或对象,以及获取当前对象。
设定分发器对象时,可以传入一个分发器类的名字,
该方法将加载类文件并创建实例。 

获取分发器对象时,首先检查是否已有一个存在,
如果没有,将创建一个默认的分发器实例。 

(set|get)Response()方法指定分发过程中使
用的响应类或对象,已经获取当前对象。设定响应对象时
,可以传入一个响应类的名字,该方法将加载类文件并创建实例。 

registerPlugin(Zend_Controller_Plugin_Abstract $plugin, $stackIndex = null)
方法允许注册一个插件对象。
通过设置可选参数$stackIndex,插件执行的顺序。 

unregisterPlugin($plugin)方法移除插件对象。
$plugin可以是一个插件对象或
者代表移除插件类的字符串。 

throwExceptions($flag)方法用来
开启或者关闭分发过程中抛出异常的能力。
默认的,异常引起并放置在响应对象中;开启throwExceptions()将覆盖这一行为。 

想知道更多信息的话,请阅读第 7.12 节 “MVC 异常”。 

returnResponse($flag)方法通知前端控制器
是否从dispatch()中返回请求对象(true),
否则自动发送响应对象(false—)。
默认的,响应对象被自动发送
(通过调用Zend_Controller_Response_Abstract::sendResponse());
开启returnResponse()将覆盖这一行为。 

返回响应对象的原因包括希望在发送响应前检查异常,
记录响应的各种属性(例如消息头)等等。 

2.3.4. 前端控制器参数
介绍里曾提到前端控制器可以用作各种控制器组件的注册表。它通过一个”param”家族的方法来做到这些。这些方法允许通过前端控制器注册任意类型的数据 —— 对象和变量,可以在分发链中的任何时候获取。这些变量被传递到路由器,分发器,以及动作控制器。这些方法包括:

setParam($name, $value)方法设定值为$value的单个参数$name。 

setParams(array $params)方法通过关联数组一次设定多个参数。 

getParam($name)方法通过$name标识符获取单个参数。 

getParams()方法一次获取整个参数列表。 

clearParams()方法可以清空一个参数(传入单个字符串标识符),
清空多个参数(传入字符串标识符数组),
清空整个参数栈(不传入参数)。 

有几个预定义的参数可供设定,它们在分发链中有特别的用途: 

useDefaultControllerAlways用来提示 分发器遇到无法分发的请求时使用默认模块的默认控制器。这默认是关闭的。 

阅读第 7.12.3 节 “可能遭遇的MVC异常”获得使用该设定的更详尽信息。 

disableOutputBuffering用来提示 
is used to hint to 分发器不使用
输出缓冲来捕捉动作控制器产生的输出。
默认的,分发器捕捉任何输出并追加到响应对象的主体内容。 

noViewRenderer用来禁用ViewRenderer。设定该参数为true可以禁用该助手。 

noErrorHandler 用来禁用错误处理器插件。设定该参数为true可以禁用该插件。 

2.3.5. 继承前端控制器
要继承前端控制器,至少需要覆盖getInstance()方法:

class My_Controller_Front extends Zend_Controller_Front
{
    public static function getInstance()
    {
        if (null === self::$_instance) {
            self::$_instance = new self();
        }

        return self::$_instance;
    }
}

覆盖getInstance()保证后面调用Zend_Controller_Front::getInstance()会返回子类的实例,而不是Zend_Controller_Front实例 —— 这对于一些可替换的路由器和视图助手非常有用。

通常不需要继承前端控制器,除非你需要增加新的功能(比如,一个插件自动加载器,或者一个方法来指定动作助手路径)。你想要改动的地方可能包括修改控制器目录的存储方式,使用的默认路由器以及分发器。

举报

相关文章推荐

Zend Framework教程-Zend_Controller_Front前端控制器

主要功能 ZendFramework的MVC实现的核心机制是通过Zend_Controller_Front前端控制器,用于初始化请求环境,处理请求,路由分发,完成响应操作,Zend_Controll...

Magento路由分发过程解析(一):在前端控制器中获取路由对象

Magento的路由系统,需要考虑到两个抽象层。 1,首先你需要了解,可能会有无数多个路由对象负责处理路由逻辑,最后只有一个路由对象能够获取并处理该请求。默认情况下,Magento拥有四个路由对象。...

我是如何成为一名python大咖的?

人生苦短,都说必须python,那么我分享下我是如何从小白成为Python资深开发者的吧。2014年我大学刚毕业..

springmvc 前端控制器,映射器,适配器,视图解析器

1.前端控制器DispatcherServlet的配置,在web.xml进行配置即可跟servlet的配置方式相同 1)contextConfigLocation配置sprimgmvc加载的配置文件...

J2EE Design Patterns 2 Front Controller(前端控制器)

表示层请求的统一入口,content retrieval, view management, and navigation。 ProblemEach view is required to provi...

spring mvc DispatcherServlet详解之前传---前端控制器架构

spring mvc DispatcherServlet详解之前传---前端控制器架构 前端控制器是整个MVC框架中最为核心的一块,它主要用来拦截符合要求的外部请求,并把请求分发到不同的控制...

【SpringMVC框架】前端控制器源代码分析

前端控制器源代码分析 虽然前面讲了一些springmvc的入门程序和配置文件中映射器和适配器的配置,但是我们作为编程人员,了解框架的部分源码还是有必要的,比如前端控制器,它是如何通过Servlet的...

openresty 前端开发轻量级MVC框架封装一(控制器篇)

通过前面几章,我们已经掌握了一些基本的开发知识,但是代码结构比较简单,缺乏统一的标准,模块化,也缺乏统一的异常处理,这一章我们主要来学习如何封装一个轻量级的MVC框架,规范以及简化开发,并且提供类似p...

存储器前端控制器多路径机制-ALUA

 ALUA即“Asymmetric Logical Unit Access(异步逻辑单元访问)”的缩写,它是前端控制器多路径机制之一。前端控制器多路径机制一定程度上决定存储的读写性能和可...
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)