当前位置:首页 > CMS教程 > 其它CMS > 列表

Zend Framework教程之动作的基类Zend_Controller_Action详解

发布:smiling 来源: PHP粉丝网  添加日期:2021-07-13 15:43:42 浏览: 评论:0 

这篇文章主要介绍了Zend Framework教程之动作的基类Zend_Controller_Action的用法,结合实例形式详细分析了动作的基类Zend_Controller_Action具体功能,使用方法与相关注意事项,需要的朋友可以参考下。

本文实例讲述了Zend Framework教程之动作的基类Zend_Controller_Action。分享给大家供大家参考,具体如下:

Zend_Controller_Action的实现

Zend Framework的动作控制器需要继承Zend_Controller_Action,Zend_Controller_Action提供了动作控制器的基本功能,具体参考如下代码:

Zend_Controller_Action_Interface

  1. <?php 
  2. interface Zend_Controller_Action_Interface 
  3.   /** 
  4.    * Class constructor 
  5.    * 
  6.    * The request and response objects should be registered with the 
  7.    * controller, as should be any additional optional arguments; these will be 
  8.    * available via {@link getRequest()}, {@link getResponse()}, and 
  9.    * {@link getInvokeArgs()}, respectively. 
  10.    * 
  11.    * When overriding the constructor, please consider this usage as a best 
  12.    * practice and ensure that each is registered appropriately; the easiest 
  13.    * way to do so is to simply call parent::__construct($request, $response, 
  14.    * $invokeArgs). 
  15.    * 
  16.    * After the request, response, and invokeArgs are set, the 
  17.    * {@link $_helper helper broker} is initialized. 
  18.    * 
  19.    * Finally, {@link init()} is called as the final action of 
  20.    * instantiation, and may be safely overridden to perform initialization 
  21.    * tasks; as a general rule, override {@link init()} instead of the 
  22.    * constructor to customize an action controller's instantiation. 
  23.    * 
  24.    * @param Zend_Controller_Request_Abstract $request 
  25.    * @param Zend_Controller_Response_Abstract $response 
  26.    * @param array $invokeArgs Any additional invocation arguments 
  27.    * @return void 
  28.    */ 
  29.   public function __construct(Zend_Controller_Request_Abstract $request
  30.                 Zend_Controller_Response_Abstract $response
  31.                 array $invokeArgs = array()); 
  32.   /** 
  33.    * Dispatch the requested action 
  34.    * 
  35.    * @param string $action Method name of action 
  36.    * @return void 
  37.    */ 
  38.   public function dispatch($action); 

Zend_Controller_Action

  1. <?php 
  2. require_once 'Zend/Controller/Action/HelperBroker.php'
  3. require_once 'Zend/Controller/Action/Interface.php'
  4. require_once 'Zend/Controller/Front.php'
  5. abstract class Zend_Controller_Action implements Zend_Controller_Action_Interface 
  6.   protected $_classMethods
  7.   protected $_delimiters
  8.   protected $_invokeArgs = array(); 
  9.   protected $_frontController
  10.   protected $_request = null; 
  11.   protected $_response = null; 
  12.   public $viewSuffix = 'phtml'
  13.   public $view
  14.   protected $_helper = null; 
  15.   public function __construct(Zend_Controller_Request_Abstract $request, Zend_Controller_Response_Abstract $responsearray $invokeArgs = array()) 
  16.   { 
  17.     $this->setRequest($request
  18.        ->setResponse($response
  19.        ->_setInvokeArgs($invokeArgs); 
  20.     $this->_helper = new Zend_Controller_Action_HelperBroker($this); 
  21.     $this->init(); 
  22.   } 
  23.   public function init() 
  24.   { 
  25.   } 
  26.   public function initView() 
  27.   { 
  28.     if (!$this->getInvokeArg('noViewRenderer') && $this->_helper->hasHelper('viewRenderer')) { 
  29.       return $this->view; 
  30.     } 
  31.     require_once 'Zend/View/Interface.php'
  32.     if (isset($this->view) && ($this->view instanceof Zend_View_Interface)) { 
  33.       return $this->view; 
  34.     } 
  35.     $request = $this->getRequest(); 
  36.     $module = $request->getModuleName(); 
  37.     $dirs  = $this->getFrontController()->getControllerDirectory(); 
  38.     if (emptyempty($module) || !isset($dirs[$module])) { 
  39.       $module = $this->getFrontController()->getDispatcher()->getDefaultModule(); 
  40.     } 
  41.     $baseDir = dirname($dirs[$module]) . DIRECTORY_SEPARATOR . 'views'
  42.     if (!file_exists($baseDir) || !is_dir($baseDir)) { 
  43.       require_once 'Zend/Controller/Exception.php'
  44.       throw new Zend_Controller_Exception('Missing base view directory ("' . $baseDir . '")'); 
  45.     } 
  46.     require_once 'Zend/View.php'
  47.     $this->view = new Zend_View(array('basePath' => $baseDir)); 
  48.     return $this->view; 
  49.   } 
  50.   public function render($action = null, $name = null, $noController = false) 
  51.   { 
  52.     if (!$this->getInvokeArg('noViewRenderer') && $this->_helper->hasHelper('viewRenderer')) { 
  53.       return $this->_helper->viewRenderer->render($action$name$noController); 
  54.     } 
  55.     $view  = $this->initView(); 
  56.     $script = $this->getViewScript($action$noController); 
  57.     $this->getResponse()->appendBody( 
  58.       $view->render($script), 
  59.       $name 
  60.     ); 
  61.   } 
  62.   public function renderScript($script$name = null) 
  63.   { 
  64.     if (!$this->getInvokeArg('noViewRenderer') && $this->_helper->hasHelper('viewRenderer')) { 
  65.       return $this->_helper->viewRenderer->renderScript($script$name); 
  66.     } 
  67.     $view = $this->initView(); 
  68.     $this->getResponse()->appendBody( 
  69.       $view->render($script), 
  70.       $name 
  71.     ); 
  72.   } 
  73.   public function getViewScript($action = null, $noController = null) 
  74.   { 
  75.     if (!$this->getInvokeArg('noViewRenderer') && $this->_helper->hasHelper('viewRenderer')) { 
  76.       $viewRenderer = $this->_helper->getHelper('viewRenderer'); 
  77.       if (null !== $noController) { 
  78.         $viewRenderer->setNoController($noController); 
  79.       } 
  80.       return $viewRenderer->getViewScript($action); 
  81.     } 
  82.     $request = $this->getRequest(); 
  83.     if (null === $action) { 
  84.       $action = $request->getActionName(); 
  85.     } elseif (!is_string($action)) { 
  86.       require_once 'Zend/Controller/Exception.php'
  87.       throw new Zend_Controller_Exception('Invalid action specifier for view render'); 
  88.     } 
  89.     if (null === $this->_delimiters) { 
  90.       $dispatcher = Zend_Controller_Front::getInstance()->getDispatcher(); 
  91.       $wordDelimiters = $dispatcher->getWordDelimiter(); 
  92.       $pathDelimiters = $dispatcher->getPathDelimiter(); 
  93.       $this->_delimiters = array_unique(array_merge($wordDelimiters, (array$pathDelimiters)); 
  94.     } 
  95.     $action = str_replace($this->_delimiters, '-'$action); 
  96.     $script = $action . '.' . $this->viewSuffix; 
  97.     if (!$noController) { 
  98.       $controller = $request->getControllerName(); 
  99.       $controller = str_replace($this->_delimiters, '-'$controller); 
  100.       $script = $controller . DIRECTORY_SEPARATOR . $script
  101.     } 
  102.     return $script
  103.   } 
  104.   public function getRequest() 
  105.   { 
  106.     return $this->_request; 
  107.   } 
  108.   public function setRequest(Zend_Controller_Request_Abstract $request
  109.   { 
  110.     $this->_request = $request
  111.     return $this
  112.   } 
  113.   public function getResponse() 
  114.   { 
  115.     return $this->_response; 
  116.   } 
  117.   public function setResponse(Zend_Controller_Response_Abstract $response
  118.   { 
  119.     $this->_response = $response
  120.     return $this
  121.   } 
  122.   protected function _setInvokeArgs(array $args = array()) 
  123.   { 
  124.     $this->_invokeArgs = $args
  125.     return $this
  126.   } 
  127.   public function getInvokeArgs() 
  128.   { 
  129.     return $this->_invokeArgs; 
  130.   } 
  131.   public function getInvokeArg($key
  132.   { 
  133.     if (isset($this->_invokeArgs[$key])) { 
  134.       return $this->_invokeArgs[$key]; 
  135.     } 
  136.     return null; 
  137.   } 
  138.   public function getHelper($helperName
  139.   { 
  140.     return $this->_helper->{$helperName}; 
  141.   } 
  142.   public function getHelperCopy($helperName
  143.   { 
  144.     return clone $this->_helper->{$helperName}; 
  145.   } 
  146.   public function setFrontController(Zend_Controller_Front $front
  147.   { 
  148.     $this->_frontController = $front
  149.     return $this
  150.   } 
  151.   public function getFrontController() 
  152.   { 
  153.     // Used cache version if found 
  154.     if (null !== $this->_frontController) { 
  155.       return $this->_frontController; 
  156.     } 
  157.     // Grab singleton instance, if class has been loaded 
  158.     if (class_exists('Zend_Controller_Front')) { 
  159.       $this->_frontController = Zend_Controller_Front::getInstance(); 
  160.       return $this->_frontController; 
  161.     } 
  162.     // Throw exception in all other cases 
  163.     require_once 'Zend/Controller/Exception.php'
  164.     throw new Zend_Controller_Exception('Front controller class has not been loaded'); 
  165.   } 
  166.   public function preDispatch() 
  167.   { 
  168.   } 
  169.   public function postDispatch() 
  170.   { 
  171.   } 
  172.   public function __call($methodName$args
  173.   { 
  174.     require_once 'Zend/Controller/Action/Exception.php'
  175.     if ('Action' == substr($methodName, -6)) { 
  176.       $action = substr($methodName, 0, strlen($methodName) - 6); 
  177.       throw new Zend_Controller_Action_Exception(sprintf('Action "%s" does not exist and was not trapped in __call()'$action), 404); 
  178.     } 
  179.     throw new Zend_Controller_Action_Exception(sprintf('Method "%s" does not exist and was not trapped in __call()'$methodName), 500); 
  180.   } 
  181.   public function dispatch($action
  182.   { 
  183.     // Notify helpers of action preDispatch state 
  184.     $this->_helper->notifyPreDispatch(); 
  185.     $this->preDispatch(); 
  186.     if ($this->getRequest()->isDispatched()) { 
  187.       if (null === $this->_classMethods) { 
  188.         $this->_classMethods = get_class_methods($this); 
  189.       } 
  190.       // If pre-dispatch hooks introduced a redirect then stop dispatch 
  191.       // @see ZF-7496 
  192.       if (!($this->getResponse()->isRedirect())) { 
  193.         // preDispatch() didn't change the action, so we can continue 
  194.         if ($this->getInvokeArg('useCaseSensitiveActions') || in_array($action$this->_classMethods)) { 
  195.           if ($this->getInvokeArg('useCaseSensitiveActions')) { 
  196.             trigger_error('Using case sensitive actions without word separators is deprecated; please do not rely on this "feature"'); 
  197.           } 
  198.           $this->$action(); 
  199.         } else { 
  200.           $this->__call($actionarray()); 
  201.         } 
  202.       } 
  203.       $this->postDispatch(); 
  204.     } 
  205.     // whats actually important here is that this action controller is 
  206.     // shutting down, regardless of dispatching; notify the helpers of this 
  207.     // state 
  208.     $this->_helper->notifyPostDispatch(); 
  209.   } 
  210.   public function run(Zend_Controller_Request_Abstract $request = null, Zend_Controller_Response_Abstract $response = null) 
  211.   { 
  212.     if (null !== $request) { 
  213.       $this->setRequest($request); 
  214.     } else { 
  215.       $request = $this->getRequest(); 
  216.     } 
  217.     if (null !== $response) { 
  218.       $this->setResponse($response); 
  219.     } 
  220.     $action = $request->getActionName(); 
  221.     if (emptyempty($action)) { 
  222.       $action = 'index'
  223.     } 
  224.     $action = $action . 'Action'
  225.     $request->setDispatched(true); 
  226.     $this->dispatch($action); 
  227.     return $this->getResponse(); 
  228.   } 
  229.   protected function _getParam($paramName$default = null) 
  230.   { 
  231.     $value = $this->getRequest()->getParam($paramName); 
  232.      if ((null === $value || '' === $value) && (null !== $default)) { 
  233.       $value = $default
  234.     } 
  235.     return $value
  236.   } 
  237.   protected function _setParam($paramName$value
  238.   { 
  239.     $this->getRequest()->setParam($paramName$value); 
  240.     return $this
  241.   } 
  242.   protected function _hasParam($paramName
  243.   { 
  244.     return null !== $this->getRequest()->getParam($paramName); 
  245.   } 
  246.   protected function _getAllParams() 
  247.   { 
  248.     return $this->getRequest()->getParams(); 
  249.   } 
  250.   final protected function _forward($action$controller = null, $module = null, array $params = null) 
  251.   { 
  252.     $request = $this->getRequest(); 
  253.     if (null !== $params) { 
  254.       $request->setParams($params); 
  255.     } 
  256.     if (null !== $controller) { 
  257.       $request->setControllerName($controller); 
  258.       // Module should only be reset if controller has been specified 
  259.       if (null !== $module) { 
  260.         $request->setModuleName($module); 
  261.       } 
  262.     } 
  263.     $request->setActionName($action
  264.         ->setDispatched(false); 
  265.   } 
  266.   protected function _redirect($urlarray $options = array()) 
  267.   { 
  268.     $this->_helper->redirector->gotoUrl($url$options); 
  269.   } 

Zend_Controller_Action提供了动作和视图的render功能,以及注册请求和响应对象,常用助手等等。

动作控制器的常用方法

在动作控制器中常用的方法和属性如下:

$this->_helper主要完成助手的相关操作例如:

只是局部控制器;当初始化加载时,对这个控制器的所有动作有效:

$this->_helper->viewRenderer->setNoRender(true);

全局:

$this->_helper->removeHelper('viewRenderer');

也是全局,但需要和本地版本协作,以便繁殖这个控制器:

Zend_Controller_Front::getInstance()->setParam('noViewRenderer', true);

通过设置ViewRenderer的noRender标记,可以简单地为一个独立的视图禁止解析(rendering):

  1. class FooController extends Zend_Controller_Action 
  2.   public function barAction() 
  3.   { 
  4.     // disable autorendering for this action only: 
  5.     $this->_helper->viewRenderer->setNoRender(); 
  6.   } 

禁止ViewRenderer的主要原因是如果你不需要视图对象或者如果你不通过视图脚本(例如,当使用动作控制器来司服网站服务协议如SOAP,XML-RPC或REST)来解析。大多数情况下,你不需要全局地禁止ViewRenderer,只选择性地在个别控制器或动作里禁止它。

请求对象和响应对象的相关操作

无数的对象和变量与对象一起注册,并且每个都有访问器方法。

请求对象:getRequest()可用来读取调用动作请求对象。

响应对象: getResponse()可用来读取收集最终响应的响应对象,一些典型的调用看起来象这样:

$this->getResponse()->setHeader('Content-Type', 'text/xml');

$this->getResponse()->appendBody($content);

调用参数:前端控制器可能把参数传给路由器、派遣器和动作控制器。为了读取这些参数,可使用getInvokeArg($key);另外,用getInvokeArgs()读取整个参数列表。

请求参数:请求对象手机请求参数,如任何_GET 或 _POST 参数,或者指定在URL的路径信息里的用户参数。为了读取这些参数,可使用_getParam($key)或_getAllParams()。也可以用_setParam()来设置请求参数;当转发到另外的动作时这很有用。

用_hasParam($key)来测试是否一个参数存在(对逻辑分支有用)。

Note: _getParam()可带有一个可选的第二个参数,如果它不是空的,就包含一个缺省的值。用它在读取值之前来消除对_hasParam() 的调用:

  1. // Use default value of 1 if id is not set 
  2. $id = $this->_getParam('id', 1); 
  3. // Instead of: 
  4. if ($this->_hasParam('id') { 
  5.   $id = $this->_getParam('id'); 
  6. else { 
  7.   $id = 1; 

视图的相关操作

Zend_Controller_Action为视图继承提供了一个初步的灵活的机制。有两个方法来完成这个:initView() 和 render();前者松散地加载$view public 属性,后者基于当前请求的动作来解析视图,它们使用目录层次来决定脚本路径。

视图初始化

initView()初始化视图对象。为了读取视图对象,render()调用initView(),但它可以在任何时候被初始化;缺省地,它用Zend_View对象来组装$view属性,但任何实现Zend_View_Interface的类可以使用,如果$view已经被初始化,它就简单地返回属性。

缺省的实现使用下面假设的目录结构:

  1. applicationOrModule/ 
  2.     controllers/ 
  3.         IndexController.php 
  4.     views/ 
  5.         scripts/ 
  6.             index/ 
  7.                 index.phtml 
  8.         helpers/ 
  9.         filters/ 

换句话说,视图脚本假定放在views/scripts/子目录,同时假定 views子目录还包含兄弟功能(助手和过滤器)。确定视图脚本名称和路径时,先以 views/scripts/作为基路径,然后加上以视图脚本对应控制器命名的目录。

解析(Rendering)视图

render() 有下列特征:has the following signature:

string render(string $action = null,

string $name = null,

bool $noController = false);

render()解析视图脚本。如果没有传递参数,它假定请求的脚本是[controller]/[action].phtml (.phtml是$viewSuffix属性的值)。为$action传递一个值将解析在[controller]子目录中的模板。为用[controller]重写,传递一个true值给$noController。最后,模板被解析到响应对象;如果你希望解析到一个在响应对象里指定的named segment,传递一个值给$name。

Note: 因为控制器和动作名字里可能包含分隔符如'_'、 '.' 和 '-',当决定视图名字时,render()把它们规格化成 '-'.在内部,它使用派遣器的字和路径分隔符来做规格化。这样,对/foo.bar/baz-bat的请求将解析脚本foo-bar/baz-bat.phtml。如果动作方法包含camelCasing,记住当决定视图脚本文件名的时候,这将变成由'-'分隔的字。

一些例子:

  1. class MyController extends Zend_Controller_Action 
  2.   public function fooAction() 
  3.   { 
  4.     // Renders my/foo.phtml 
  5.     $this->render(); 
  6.     // Renders my/bar.phtml 
  7.     $this->render('bar'); 
  8.     // Renders baz.phtml 
  9.     $this->render('baz', null, true); 
  10.     // Renders my/login.phtml to the 'form' segment of the 
  11.     // response object 
  12.     $this->render('login''form'); 
  13.     // Renders site.phtml to the 'page' segment of the response 
  14.     // object; does not use the 'my/' subirectory 
  15.     $this->render('site''page', true); 
  16.   } 
  17.   public function bazBatAction() 
  18.   { 
  19.     // Renders my/baz-bat.phtml 
  20.     $this->render(); 
  21.   } 

其它

_forward($action, $controller = null, $module = null, array $params = null) :执行另外一个动作。如果在preDispatch()里调用,当前请求的动作将被跳过来支持新的动作。否则,在当前动作被处理之后,在_forward()请求的动作将被执行。

_redirect($url, array $options = array()):重定向到另外一个地方,这个方法用URL和一组可选的选项,缺省地,它执行HTTP 302 重定向。

选项可包括一个或多个下面这些:

exit:是否立即退出。如果被请求,它将干净地关闭任何打开的会话和执行重定向。

可以用setRedirectExit()访问器在控制器里全局地设置这个选项。

prependBase:是否预先考虑基础URL和URL提供的请求对象一起注册。

使用setRedirectPrependBase()访问器,在控制器里全局地设置这个选项。

code:在重定向时要用什么HTTP代码。缺省使用302;可以用从301到306之间的任何代码。

使用setRedirectCode()访问器,在控制器里全局地设置这个选项。

扩展自定义Zend_Controller_Action

为了创建动作控制器,设计上,Zend_Controller_Action 必须被继承。至少,需要定义控制器可能调用的动作方法。

除了为web应用程序创建有用的函数外,你可能发现在不同的控制器里重复同样的设置和实用方法;如果这样,创建一个继承(extends)Zend_Controller_Action 的基础类可能会解决问题。

Example #1 如何处理不存在的动作

如果控制器的请求包括一个未定义的动作方法,Zend_Controller_Action::__call()将被调用。__call()当然是PHP中用来重载方法的魔术方法。

缺省地,这个方法抛出一个Zend_Controller_Action_Exception 来表明在控制器里没有发现要求的方法。如果要求的方法以'Action'结尾,就假设一个动作被请求并且不存在;这样的错误导致带有代码为 404 的异常。所有其它方法导致带有代码为 500 的异常。这使你很容易地在错误句柄里区分是页面没有发现还是程序错误。

如果想执行其它操作,你应该重写这个函数。例如,如果你想显示错误信息,可以象下面这样来写:

  1. class MyController extends Zend_Controller_Action 
  2.   public function __call($method$args
  3.   { 
  4.     if ('Action' == substr($method, -6)) { 
  5.       // If the action method was not found, render the error 
  6.       // template 
  7.       return $this->render('error'); 
  8.     } 
  9.     // all other methods throw an exception 
  10.     throw new Exception('Invalid method "' 
  11.               . $method 
  12.               . '" called'
  13.               500); 
  14.   } 

另外的可能性就是你可能想转发到缺省控制页面:

  1. class MyController extends Zend_Controller_Action 
  2.   public function indexAction() 
  3.   { 
  4.     $this->render(); 
  5.   } 
  6.   public function __call($method$args
  7.   { 
  8.     if ('Action' == substr($method, -6)) { 
  9.       // If the action method was not found, forward to the 
  10.       // index action 
  11.       return $this->_forward('index'); 
  12.     } 
  13.     // all other methods throw an exception 
  14.     throw new Exception('Invalid method "' 
  15.               . $method 
  16.               . '" called'
  17.               500); 
  18.   } 

为了定制控制器,除了重写__call()以外,本章前面说涉及的初始化、实用程序、访问器、视图和派遣钩子等方法都可以被重写,作为例子,如果把视图对象保存到注册表里,你可能想用象下面的代码来修改initView():

  1. abstract class My_Base_Controller extends Zend_Controller_Action 
  2.   public function initView() 
  3.   { 
  4.     if (null === $this->view) { 
  5.       if (Zend_Registry::isRegistered('view')) { 
  6.         $this->view = Zend_Registry::get('view'); 
  7.       } else { 
  8.         $this->view = new Zend_View(); 
  9.         $this->view->setBasePath(dirname(__FILE__) . '/../views'); 
  10.       } 
  11.     } 
  12.     return $this->view; 
  13.   } 
  14. }

Tags: Zend_Controller_Action

分享到: