插件 简介 控制器(Controller)结构包含一个可以在控制器周期内确定事件发生时调用用户代码的插件系统。 前端控制器(Front controller)使用插件 broker 作为用户插件注册,同时插件 broker 确保前端控制器中注册的每个插件都在事件发生时调用相应的事件方法。 事件方法定义在虚类 Zend_Controller_Plugin_Abstract,用户插件应当从这个类继承: routeStartup()Zend_Controller_Front 向注册的 路由器 发送请求前被调用。 routeShutdown()路由器 完成请求的路由后被调用。 dispatchLoopStartup()Zend_Controller_Front 进入其分发循环(dispatch loop)前被调用。 preDispatch() 在动作由 分发器 分发前被调用。该回调方法允许代理或者过滤行为。通过修改请求和重设分发标志位(利用 Zend_Controller_Request_Abstract::setDispatched(false) )当前动作可以跳过或者被替换。 postDispatch() 在动作由 分发器 分发后被调用。该回调方法允许代理或者过滤行为。通过修改请求和重设分发标志位(利用 Zend_Controller_Request_Abstract::setDispatched(false) )可以指定新动作进行分发。 dispatchLoopShutdown()Zend_Controller_Front 推出其分发循环后调用。 编写插件 只需要包含并扩展抽象类 Zend_Controller_Plugin_Abstract 即可编写插件类。 Zend_Controller_Plugin_Abstract 的全部方法都不是抽象的, 这意味着插件类并不是一定要去实现前面列出的每一个事件方法。 插件的开发者只要实现需要用到的方法即可。 Zend_Controller_Plugin_Abstract 也可以通过调用 getRequest()getResponse() 方法从控制器中分别获取 request 对象和 response 对象. 使用插件 可以使用 Zend_Controller_Front::registerPlugin() 在任何时候注册插件类。 下面的代码片段说明了如何在控制器链条中使用插件。 getResponse()->appendBody("

routeStartup() called

\n"); } public function routeShutdown(Zend_Controller_Request_Abstract $request) { $this->getResponse()->appendBody("

routeShutdown() called

\n"); } public function dispatchLoopStartup(Zend_Controller_Request_Abstract $request) { $this->getResponse()->appendBody("

dispatchLoopStartup() called

\n"); } public function preDispatch(Zend_Controller_Request_Abstract $request) { $this->getResponse()->appendBody("

preDispatch() called

\n"); } public function postDispatch(Zend_Controller_Request_Abstract $request) { $this->getResponse()->appendBody("

postDispatch() called

\n"); } public function dispatchLoopShutdown() { $this->getResponse()->appendBody("

dispatchLoopShutdown() called

\n"); } } $front = Zend_Controller_Front::getInstance(); $front->setControllerDirectory('/path/to/controllers') ->setRouter(new Zend_Controller_Router_Rewrite()) ->registerPlugin(new MyPlugin()); $front->dispatch(); ]]>
假设没有动作产生任何输出,而只有一个动作被调用,前面演示的插件仍然会产生下面的输出: routeStartup() called

routeShutdown() called

dispatchLoopStartup() called

preDispatch() called

postDispatch() called

dispatchLoopShutdown() called

]]>
插件可以在前端控制器(Front controller)执行的任何时候被被注册, 如果一个事件在注册时已经完成,则这个事件对应的事件方法不会被触发。
获取和控制插件 有时,可能需要取消注册或者获取一个插件。下面列出的前端控制器中的方法可以实现这个功能: getPlugin($class) 允许获取指定类名的一个插件。 如果没有插件匹配,将返回 false。如果有多个指定类的插件被注册,则返回一个数组。 getPlugins() 返回全部插件。 unregisterPlugin($plugin) 允许从插件列表中移除一个插件。 传递一个插件件对象,或者需要移除的插件的类名。如果传递类名,任何该类的插件都将被移除。 包含在标准发行包中的插件 Zend Framework 在其标准发行包中包含错误处理插件。