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

Zend Framework教程之路由功能Zend_Controller_Router详解

发布:smiling 来源: PHP粉丝网  添加日期:2021-07-13 16:01:39 浏览: 评论:0 

本文实例讲述了Zend Framework教程之路由功能Zend_Controller_Router用法,分享给大家供大家参考,具体如下:

Zend Framework的路由提供了两个主要功能路由和创建路由。

Zend_Controller_Router的Route类和相应Route目录下的类定义常见的路由操作。

接口Zend_Controller_Router_Interface,类Zend_Controller_Router_Abstract和Zend_Controller_Router_Rewrite完成了基本的路由,创建路由,删除路由的功能。

  1. └── Router 
  2.     ├── Abstract.php 
  3.     ├── Exception.php 
  4.     ├── Interface.php 
  5.     ├── Rewrite.php 
  6.     ├── Route 
  7.     │   ├── Abstract.php 
  8.     │   ├── Chain.php 
  9.     │   ├── Hostname.php 
  10.     │   ├── Interface.php 
  11.     │   ├── Module.php 
  12.     │   ├── Regex.php 
  13.     │   └── Static.php 
  14.     └── Route.php 

Zend_Controller_Router路由功能的实现

Zend_Controller_Router_Interface

  1. <?php 
  2. interface Zend_Controller_Router_Interface 
  3.   /** 
  4.    * Processes a request and sets its controller and action. If 
  5.    * no route was possible, an exception is thrown. 
  6.    * 
  7.    * @param Zend_Controller_Request_Abstract 
  8.    * @throws Zend_Controller_Router_Exception 
  9.    * @return Zend_Controller_Request_Abstract|boolean 
  10.    */ 
  11.   public function route(Zend_Controller_Request_Abstract $dispatcher); 
  12.   /** 
  13.    * Generates a URL path that can be used in URL creation, redirection, etc. 
  14.    * 
  15.    * May be passed user params to override ones from URI, Request or even defaults. 
  16.    * If passed parameter has a value of null, it's URL variable will be reset to 
  17.    * default. 
  18.    * 
  19.    * If null is passed as a route name assemble will use the current Route or 'default' 
  20.    * if current is not yet set. 
  21.    * 
  22.    * Reset is used to signal that all parameters should be reset to it's defaults. 
  23.    * Ignoring all URL specified values. User specified params still get precedence. 
  24.    * 
  25.    * Encode tells to url encode resulting path parts. 
  26.    * 
  27.    * @param array $userParams Options passed by a user used to override parameters 
  28.    * @param mixed $name The name of a Route to use 
  29.    * @param bool $reset Whether to reset to the route defaults ignoring URL params 
  30.    * @param bool $encode Tells to encode URL parts on output 
  31.    * @throws Zend_Controller_Router_Exception 
  32.    * @return string Resulting URL path 
  33.    */ 
  34.   public function assemble($userParams$name = null, $reset = false, $encode = true); 
  35.   /** 
  36.    * Retrieve Front Controller 
  37.    * 
  38.    * @return Zend_Controller_Front 
  39.    */ 
  40.   public function getFrontController(); 
  41.   /** 
  42.    * Set Front Controller 
  43.    * 
  44.    * @param Zend_Controller_Front $controller 
  45.    * @return Zend_Controller_Router_Interface 
  46.    */ 
  47.   public function setFrontController(Zend_Controller_Front $controller); 
  48.   /** 
  49.    * Add or modify a parameter with which to instantiate any helper objects 
  50.    * 
  51.    * @param string $name 
  52.    * @param mixed $param 
  53.    * @return Zend_Controller_Router_Interface 
  54.    */ 
  55.   public function setParam($name$value); 
  56.   /** 
  57.    * Set an array of a parameters to pass to helper object constructors 
  58.    * 
  59.    * @param array $params 
  60.    * @return Zend_Controller_Router_Interface 
  61.    */ 
  62.   public function setParams(array $params); 
  63.   /** 
  64.    * Retrieve a single parameter from the controller parameter stack 
  65.    * 
  66.    * @param string $name 
  67.    * @return mixed 
  68.    */ 
  69.   public function getParam($name); 
  70.   /** 
  71.    * Retrieve the parameters to pass to helper object constructors 
  72.    * 
  73.    * @return array 
  74.    */ 
  75.   public function getParams(); 
  76.   /** 
  77.    * Clear the controller parameter stack 
  78.    * 
  79.    * By default, clears all parameters. If a parameter name is given, clears 
  80.    * only that parameter; if an array of parameter names is provided, clears 
  81.    * each. 
  82.    * 
  83.    * @param null|string|array single key or array of keys for params to clear 
  84.    * @return Zend_Controller_Router_Interface 
  85.    */ 
  86.   public function clearParams($name = null); 

Zend_Controller_Router_Abstract

  1. <?php 
  2. /** Zend_Controller_Router_Interface */ 
  3. require_once 'Zend/Controller/Router/Interface.php'
  4. abstract class Zend_Controller_Router_Abstract implements Zend_Controller_Router_Interface 
  5.   /** 
  6.    * URI delimiter 
  7.    */ 
  8.   const URI_DELIMITER = '/'
  9.   /** 
  10.    * Front controller instance 
  11.    * @var Zend_Controller_Front 
  12.    */ 
  13.   protected $_frontController
  14.   /** 
  15.    * Array of invocation parameters to use when instantiating action 
  16.    * controllers 
  17.    * @var array 
  18.    */ 
  19.   protected $_invokeParams = array(); 
  20.   /** 
  21.    * Constructor 
  22.    * 
  23.    * @param array $params 
  24.    * @return void 
  25.    */ 
  26.   public function __construct(array $params = array()) 
  27.   { 
  28.     $this->setParams($params); 
  29.   } 
  30.   /** 
  31.    * Add or modify a parameter to use when instantiating an action controller 
  32.    * 
  33.    * @param string $name 
  34.    * @param mixed $value 
  35.    * @return Zend_Controller_Router 
  36.    */ 
  37.   public function setParam($name$value
  38.   { 
  39.     $name = (string) $name
  40.     $this->_invokeParams[$name] = $value
  41.     return $this
  42.   } 
  43.   /** 
  44.    * Set parameters to pass to action controller constructors 
  45.    * 
  46.    * @param array $params 
  47.    * @return Zend_Controller_Router 
  48.    */ 
  49.   public function setParams(array $params
  50.   { 
  51.     $this->_invokeParams = array_merge($this->_invokeParams, $params); 
  52.     return $this
  53.   } 
  54.   /** 
  55.    * Retrieve a single parameter from the controller parameter stack 
  56.    * 
  57.    * @param string $name 
  58.    * @return mixed 
  59.    */ 
  60.   public function getParam($name
  61.   { 
  62.     if(isset($this->_invokeParams[$name])) { 
  63.       return $this->_invokeParams[$name]; 
  64.     } 
  65.     return null; 
  66.   } 
  67.   /** 
  68.    * Retrieve action controller instantiation parameters 
  69.    * 
  70.    * @return array 
  71.    */ 
  72.   public function getParams() 
  73.   { 
  74.     return $this->_invokeParams; 
  75.   } 
  76.   /** 
  77.    * Clear the controller parameter stack 
  78.    * 
  79.    * By default, clears all parameters. If a parameter name is given, clears 
  80.    * only that parameter; if an array of parameter names is provided, clears 
  81.    * each. 
  82.    * 
  83.    * @param null|string|array single key or array of keys for params to clear 
  84.    * @return Zend_Controller_Router 
  85.    */ 
  86.   public function clearParams($name = null) 
  87.   { 
  88.     if (null === $name) { 
  89.       $this->_invokeParams = array(); 
  90.     } elseif (is_string($name) && isset($this->_invokeParams[$name])) { 
  91.       unset($this->_invokeParams[$name]); 
  92.     } elseif (is_array($name)) { 
  93.       foreach ($name as $key) { 
  94.         if (is_string($key) && isset($this->_invokeParams[$key])) { 
  95.           unset($this->_invokeParams[$key]); 
  96.         } 
  97.       } 
  98.     } 
  99.     return $this
  100.   } 
  101.   /** 
  102.    * Retrieve Front Controller 
  103.    * 
  104.    * @return Zend_Controller_Front 
  105.    */ 
  106.   public function getFrontController() 
  107.   { 
  108.     // Used cache version if found 
  109.     if (null !== $this->_frontController) { 
  110.       return $this->_frontController; 
  111.     } 
  112.     require_once 'Zend/Controller/Front.php'
  113.     $this->_frontController = Zend_Controller_Front::getInstance(); 
  114.     return $this->_frontController; 
  115.   } 
  116.   /** 
  117.    * Set Front Controller 
  118.    * 
  119.    * @param Zend_Controller_Front $controller 
  120.    * @return Zend_Controller_Router_Interface 
  121.    */ 
  122.   public function setFrontController(Zend_Controller_Front $controller
  123.   { 
  124.     $this->_frontController = $controller
  125.     return $this
  126.   } 

Zend_Controller_Router_Rewrite

  1. <?php 
  2. /** Zend_Controller_Router_Abstract */ 
  3. require_once 'Zend/Controller/Router/Abstract.php'
  4. /** Zend_Controller_Router_Route */ 
  5. require_once 'Zend/Controller/Router/Route.php'
  6. class Zend_Controller_Router_Rewrite extends Zend_Controller_Router_Abstract 
  7.   /** 
  8.    * Whether or not to use default routes 
  9.    * 
  10.    * @var boolean 
  11.    */ 
  12.   protected $_useDefaultRoutes = true; 
  13.   /** 
  14.    * Array of routes to match against 
  15.    * 
  16.    * @var array 
  17.    */ 
  18.   protected $_routes = array(); 
  19.   /** 
  20.    * Currently matched route 
  21.    * 
  22.    * @var Zend_Controller_Router_Route_Interface 
  23.    */ 
  24.   protected $_currentRoute = null; 
  25.   /** 
  26.    * Global parameters given to all routes 
  27.    * 
  28.    * @var array 
  29.    */ 
  30.   protected $_globalParams = array(); 
  31.   /** 
  32.    * Separator to use with chain names 
  33.    * 
  34.    * @var string 
  35.    */ 
  36.   protected $_chainNameSeparator = '-'
  37.   /** 
  38.    * Determines if request parameters should be used as global parameters 
  39.    * inside this router. 
  40.    * 
  41.    * @var boolean 
  42.    */ 
  43.   protected $_useCurrentParamsAsGlobal = false; 
  44.   /** 
  45.    * Add default routes which are used to mimic basic router behaviour 
  46.    * 
  47.    * @return Zend_Controller_Router_Rewrite 
  48.    */ 
  49.   public function addDefaultRoutes() 
  50.   { 
  51.     if (!$this->hasRoute('default')) { 
  52.       $dispatcher = $this->getFrontController()->getDispatcher(); 
  53.       $request = $this->getFrontController()->getRequest(); 
  54.       require_once 'Zend/Controller/Router/Route/Module.php'
  55.       $compat = new Zend_Controller_Router_Route_Module(array(), $dispatcher$request); 
  56.       $this->_routes = array('default' => $compat) + $this->_routes; 
  57.     } 
  58.     return $this
  59.   } 
  60.   /** 
  61.    * Add route to the route chain 
  62.    * 
  63.    * If route contains method setRequest(), it is initialized with a request object 
  64.    * 
  65.    * @param string                 $name    Name of the route 
  66.    * @param Zend_Controller_Router_Route_Interface $route   Instance of the route 
  67.    * @return Zend_Controller_Router_Rewrite 
  68.    */ 
  69.   public function addRoute($name, Zend_Controller_Router_Route_Interface $route
  70.   { 
  71.     if (method_exists($route'setRequest')) { 
  72.       $route->setRequest($this->getFrontController()->getRequest()); 
  73.     } 
  74.     $this->_routes[$name] = $route
  75.     return $this
  76.   } 
  77.   /** 
  78.    * Add routes to the route chain 
  79.    * 
  80.    * @param array $routes Array of routes with names as keys and routes as values 
  81.    * @return Zend_Controller_Router_Rewrite 
  82.    */ 
  83.   public function addRoutes($routes) { 
  84.     foreach ($routes as $name => $route) { 
  85.       $this->addRoute($name$route); 
  86.     } 
  87.     return $this
  88.   } 
  89.   /** 
  90.    * Create routes out of Zend_Config configuration 
  91.    * 
  92.    * Example INI: 
  93.    * routes.archive.route = "archive/:year/*" 
  94.    * routes.archive.defaults.controller = archive 
  95.    * routes.archive.defaults.action = show 
  96.    * routes.archive.defaults.year = 2000 
  97.    * routes.archive.reqs.year = "\d+" 
  98.    * 
  99.    * routes.news.type = "Zend_Controller_Router_Route_Static" 
  100.    * routes.news.route = "news" 
  101.    * routes.news.defaults.controller = "news" 
  102.    * routes.news.defaults.action = "list" 
  103.    * 
  104.    * And finally after you have created a Zend_Config with above ini: 
  105.    * $router = new Zend_Controller_Router_Rewrite(); 
  106.    * $router->addConfig($config, 'routes'); 
  107.    * 
  108.    * @param Zend_Config $config Configuration object 
  109.    * @param string   $section Name of the config section containing route's definitions 
  110.    * @throws Zend_Controller_Router_Exception 
  111.    * @return Zend_Controller_Router_Rewrite 
  112.    */ 
  113.   public function addConfig(Zend_Config $config$section = null) 
  114.   { 
  115.     if ($section !== null) { 
  116.       if ($config->{$section} === null) { 
  117.         require_once 'Zend/Controller/Router/Exception.php'
  118.         throw new Zend_Controller_Router_Exception("No route configuration in section '{$section}'"); 
  119.       } 
  120.       $config = $config->{$section}; 
  121.     } 
  122.     foreach ($config as $name => $info) { 
  123.       $route = $this->_getRouteFromConfig($info); 
  124.       if ($route instanceof Zend_Controller_Router_Route_Chain) { 
  125.         if (!isset($info->chain)) { 
  126.           require_once 'Zend/Controller/Router/Exception.php'
  127.           throw new Zend_Controller_Router_Exception("No chain defined"); 
  128.         } 
  129.         if ($info->chain instanceof Zend_Config) { 
  130.           $childRouteNames = $info->chain; 
  131.         } else { 
  132.           $childRouteNames = explode(','$info->chain); 
  133.         } 
  134.         foreach ($childRouteNames as $childRouteName) { 
  135.           $childRoute = $this->getRoute(trim($childRouteName)); 
  136.           $route->chain($childRoute); 
  137.         } 
  138.         $this->addRoute($name$route); 
  139.       } elseif (isset($info->chains) && $info->chains instanceof Zend_Config) { 
  140.         $this->_addChainRoutesFromConfig($name$route$info->chains); 
  141.       } else { 
  142.         $this->addRoute($name$route); 
  143.       } 
  144.     } 
  145.     return $this
  146.   } 
  147.   /** 
  148.    * Get a route frm a config instance 
  149.    * 
  150.    * @param Zend_Config $info 
  151.    * @return Zend_Controller_Router_Route_Interface 
  152.    */ 
  153.   protected function _getRouteFromConfig(Zend_Config $info
  154.   { 
  155.     $class = (isset($info->type)) ? $info->type : 'Zend_Controller_Router_Route'
  156.     if (!class_exists($class)) { 
  157.       require_once 'Zend/Loader.php'
  158.       Zend_Loader::loadClass($class); 
  159.     } 
  160.     $route = call_user_func(array($class'getInstance'), $info); 
  161.     if (isset($info->abstract) && $info->abstract && method_exists($route'isAbstract')) { 
  162.       $route->isAbstract(true); 
  163.     } 
  164.     return $route
  165.   } 
  166.   /** 
  167.    * Add chain routes from a config route 
  168.    * 
  169.    * @param string                 $name 
  170.    * @param Zend_Controller_Router_Route_Interface $route 
  171.    * @param Zend_Config              $childRoutesInfo 
  172.    * @return void 
  173.    */ 
  174.   protected function _addChainRoutesFromConfig($name
  175.                          Zend_Controller_Router_Route_Interface $route
  176.                          Zend_Config $childRoutesInfo
  177.   { 
  178.     foreach ($childRoutesInfo as $childRouteName => $childRouteInfo) { 
  179.       if (is_string($childRouteInfo)) { 
  180.         $childRouteName = $childRouteInfo
  181.         $childRoute   = $this->getRoute($childRouteName); 
  182.       } else { 
  183.         $childRoute = $this->_getRouteFromConfig($childRouteInfo); 
  184.       } 
  185.       if ($route instanceof Zend_Controller_Router_Route_Chain) { 
  186.         $chainRoute = clone $route
  187.         $chainRoute->chain($childRoute); 
  188.       } else { 
  189.         $chainRoute = $route->chain($childRoute); 
  190.       } 
  191.       $chainName = $name . $this->_chainNameSeparator . $childRouteName
  192.       if (isset($childRouteInfo->chains)) { 
  193.         $this->_addChainRoutesFromConfig($chainName$chainRoute$childRouteInfo->chains); 
  194.       } else { 
  195.         $this->addRoute($chainName$chainRoute); 
  196.       } 
  197.     } 
  198.   } 
  199.   /** 
  200.    * Remove a route from the route chain 
  201.    * 
  202.    * @param string $name Name of the route 
  203.    * @throws Zend_Controller_Router_Exception 
  204.    * @return Zend_Controller_Router_Rewrite 
  205.    */ 
  206.   public function removeRoute($name
  207.   { 
  208.     if (!isset($this->_routes[$name])) { 
  209.       require_once 'Zend/Controller/Router/Exception.php'
  210.       throw new Zend_Controller_Router_Exception("Route $name is not defined"); 
  211.     } 
  212.     unset($this->_routes[$name]); 
  213.     return $this
  214.   } 
  215.   /** 
  216.    * Remove all standard default routes 
  217.    * 
  218.    * @param Zend_Controller_Router_Route_Interface Route 
  219.    * @return Zend_Controller_Router_Rewrite 
  220.    */ 
  221.   public function removeDefaultRoutes() 
  222.   { 
  223.     $this->_useDefaultRoutes = false; 
  224.     return $this
  225.   } 
  226.   /** 
  227.    * Check if named route exists 
  228.    * 
  229.    * @param string $name Name of the route 
  230.    * @return boolean 
  231.    */ 
  232.   public function hasRoute($name
  233.   { 
  234.     return isset($this->_routes[$name]); 
  235.   } 
  236.   /** 
  237.    * Retrieve a named route 
  238.    * 
  239.    * @param string $name Name of the route 
  240.    * @throws Zend_Controller_Router_Exception 
  241.    * @return Zend_Controller_Router_Route_Interface Route object 
  242.    */ 
  243.   public function getRoute($name
  244.   { 
  245.     if (!isset($this->_routes[$name])) { 
  246.       require_once 'Zend/Controller/Router/Exception.php'
  247.       throw new Zend_Controller_Router_Exception("Route $name is not defined"); 
  248.     } 
  249.     return $this->_routes[$name]; 
  250.   } 
  251.   /** 
  252.    * Retrieve a currently matched route 
  253.    * 
  254.    * @throws Zend_Controller_Router_Exception 
  255.    * @return Zend_Controller_Router_Route_Interface Route object 
  256.    */ 
  257.   public function getCurrentRoute() 
  258.   { 
  259.     if (!isset($this->_currentRoute)) { 
  260.       require_once 'Zend/Controller/Router/Exception.php'
  261.       throw new Zend_Controller_Router_Exception("Current route is not defined"); 
  262.     } 
  263.     return $this->getRoute($this->_currentRoute); 
  264.   } 
  265.   /** 
  266.    * Retrieve a name of currently matched route 
  267.    * 
  268.    * @throws Zend_Controller_Router_Exception 
  269.    * @return Zend_Controller_Router_Route_Interface Route object 
  270.    */ 
  271.   public function getCurrentRouteName() 
  272.   { 
  273.     if (!isset($this->_currentRoute)) { 
  274.       require_once 'Zend/Controller/Router/Exception.php'
  275.       throw new Zend_Controller_Router_Exception("Current route is not defined"); 
  276.     } 
  277.     return $this->_currentRoute; 
  278.   } 
  279.   /** 
  280.    * Retrieve an array of routes added to the route chain 
  281.    * 
  282.    * @return array All of the defined routes 
  283.    */ 
  284.   public function getRoutes() 
  285.   { 
  286.     return $this->_routes; 
  287.   } 
  288.   /** 
  289.    * Find a matching route to the current PATH_INFO and inject 
  290.    * returning values to the Request object. 
  291.    * 
  292.    * @throws Zend_Controller_Router_Exception 
  293.    * @return Zend_Controller_Request_Abstract Request object 
  294.    */ 
  295.   public function route(Zend_Controller_Request_Abstract $request
  296.   { 
  297.     if (!$request instanceof Zend_Controller_Request_Http) { 
  298.       require_once 'Zend/Controller/Router/Exception.php'
  299.       throw new Zend_Controller_Router_Exception('Zend_Controller_Router_Rewrite requires a Zend_Controller_Request_Http-based request object'); 
  300.     } 
  301.     if ($this->_useDefaultRoutes) { 
  302.       $this->addDefaultRoutes(); 
  303.     } 
  304.     // Find the matching route 
  305.     $routeMatched = false; 
  306.     foreach (array_reverse($this->_routes, true) as $name => $route) { 
  307.       // TODO: Should be an interface method. Hack for 1.0 BC 
  308.       if (method_exists($route'isAbstract') && $route->isAbstract()) { 
  309.         continue
  310.       } 
  311.       // TODO: Should be an interface method. Hack for 1.0 BC 
  312.       if (!method_exists($route'getVersion') || $route->getVersion() == 1) { 
  313.         $match = $request->getPathInfo(); 
  314.       } else { 
  315.         $match = $request
  316.       } 
  317.       if ($params = $route->match($match)) { 
  318.         $this->_setRequestParams($request$params); 
  319.         $this->_currentRoute = $name
  320.         $routeMatched    = true; 
  321.         break
  322.       } 
  323.     } 
  324.      if (!$routeMatched) { 
  325.        require_once 'Zend/Controller/Router/Exception.php'
  326.        throw new Zend_Controller_Router_Exception('No route matched the request', 404); 
  327.      } 
  328.     if($this->_useCurrentParamsAsGlobal) { 
  329.       $params = $request->getParams(); 
  330.       foreach($params as $param => $value) { 
  331.         $this->setGlobalParam($param$value); 
  332.       } 
  333.     } 
  334.     return $request
  335.   } 
  336.   protected function _setRequestParams($request$params
  337.   { 
  338.     foreach ($params as $param => $value) { 
  339.       $request->setParam($param$value); 
  340.       if ($param === $request->getModuleKey()) { 
  341.         $request->setModuleName($value); 
  342.       } 
  343.       if ($param === $request->getControllerKey()) { 
  344.         $request->setControllerName($value); 
  345.       } 
  346.       if ($param === $request->getActionKey()) { 
  347.         $request->setActionName($value); 
  348.       } 
  349.     } 
  350.   } 
  351.   /** 
  352.    * Generates a URL path that can be used in URL creation, redirection, etc. 
  353.    * 
  354.    * @param array $userParams Options passed by a user used to override parameters 
  355.    * @param mixed $name The name of a Route to use 
  356.    * @param bool $reset Whether to reset to the route defaults ignoring URL params 
  357.    * @param bool $encode Tells to encode URL parts on output 
  358.    * @throws Zend_Controller_Router_Exception 
  359.    * @return string Resulting absolute URL path 
  360.    */ 
  361.   public function assemble($userParams$name = null, $reset = false, $encode = true) 
  362.   { 
  363.     if (!is_array($userParams)) { 
  364.       require_once 'Zend/Controller/Router/Exception.php'
  365.       throw new Zend_Controller_Router_Exception('userParams must be an array'); 
  366.     } 
  367.     if ($name == null) { 
  368.       try { 
  369.         $name = $this->getCurrentRouteName(); 
  370.       } catch (Zend_Controller_Router_Exception $e) { 
  371.         $name = 'default'
  372.       } 
  373.     } 
  374.     // Use UNION (+) in order to preserve numeric keys 
  375.     $params = $userParams + $this->_globalParams; 
  376.     $route = $this->getRoute($name); 
  377.     $url  = $route->assemble($params$reset$encode); 
  378.     if (!preg_match('|^[a-z]+://|'$url)) { 
  379.       $url = rtrim($this->getFrontController()->getBaseUrl(), self::URI_DELIMITER) . self::URI_DELIMITER . $url
  380.     } 
  381.     return $url
  382.   } 
  383.   /** 
  384.    * Set a global parameter 
  385.    * 
  386.    * @param string $name 
  387.    * @param mixed $value 
  388.    * @return Zend_Controller_Router_Rewrite 
  389.    */ 
  390.   public function setGlobalParam($name$value
  391.   { 
  392.     $this->_globalParams[$name] = $value
  393.     return $this
  394.   } 
  395.   /** 
  396.    * Set the separator to use with chain names 
  397.    * 
  398.    * @param string $separator The separator to use 
  399.    * @return Zend_Controller_Router_Rewrite 
  400.    */ 
  401.   public function setChainNameSeparator($separator) { 
  402.     $this->_chainNameSeparator = $separator
  403.     return $this
  404.   } 
  405.   /** 
  406.    * Get the separator to use for chain names 
  407.    * 
  408.    * @return string 
  409.    */ 
  410.   public function getChainNameSeparator() { 
  411.     return $this->_chainNameSeparator; 
  412.   } 
  413.   /** 
  414.    * Determines/returns whether to use the request parameters as global parameters. 
  415.    * 
  416.    * @param boolean|null $use 
  417.    *      Null/unset when you want to retrieve the current state. 
  418.    *      True when request parameters should be global, false otherwise 
  419.    * @return boolean|Zend_Controller_Router_Rewrite 
  420.    *       Returns a boolean if first param isn't set, returns an 
  421.    *       instance of Zend_Controller_Router_Rewrite otherwise. 
  422.    * 
  423.    */ 
  424.   public function useRequestParametersAsGlobal($use = null) { 
  425.     if($use === null) { 
  426.       return $this->_useCurrentParamsAsGlobal; 
  427.     } 
  428.     $this->_useCurrentParamsAsGlobal = (bool) $use
  429.     return $this
  430.   } 

添加路由的操作方法

  1. public function addRoute($name, Zend_Controller_Router_Route_Interface $route
  2.  
  3. public function addRoutes($routes
  4.  
  5. $router = $ctrl->getRouter(); // returns a rewrite router by default 
  6. $router->addRoute('user'
  7.          new Zend_Controller_Router_Route('user/:username')); 

addRoute的第一个参数是路由名。第二个参数是路由自己,路由名最普通的用法是通过Zend_View_Url助手的方法:

"<?= $this->url(array('username' => 'martel'), 'user') ?>">Martel</a>

它将导致在 href: user/martel.

路由是一个简单的过程,这个过程通过所有提供的路由和匹配它的当前请求的URI定义来迭代。当一个正匹配被发现,变量值从路由实例返回并注入到Zend_Controller_Request对象以备将来在派遣器和用户创建的控制器中使用。如果是负匹配,在链中的下个路由被检查。

Note: 倒序匹配

用倒序来匹配路由确保最通用的路由被首先定义。

Note: 返回的值

从路由返回的值来自于URL参数或用于定义的缺省值。这些变量以后可通过Zend_Controller_Request::getParam() 或 Zend_Controller_Action::_getParam() 方法来访问。

有三个特殊的变量可用于你的路由-'module'、 'controller' 和 'action'。这些特殊的变量被Zend_Controller_Dispatcher用来找出控制器和动作然后派遣过去。

Note: 特殊变量

如果你选择通过 setControllerKey 和 setActionKey方法的方式来改变缺省值,这些特殊变量的名字可能会不同。

缺省路由

Zend_Controller_Router_Rewrite 和缺省路由一起预先配置,它将以controller/action的形式匹配URIs。另外,模块名可以被指定作为第一个路径参数,允许这种module/controller/action形式的URIs。最后,它也将缺省地匹配任何另外的追加到URI的参数-controller/action/var1/value1/var2/value2。

一些路由如何匹配的例子:

  1. // Assuming the following: 
  2. $ctrl->setControllerDirectory( 
  3.   array
  4.     'default' => '/path/to/default/controllers'
  5.     'news'  => '/path/to/news/controllers'
  6.     'blog'  => '/path/to/blog/controllers' 
  7.   ) 
  8. ); 
  9. Module only: 
  10. http://example/news 
  11.   module == news 
  12. Invalid module maps to controller name: 
  13. http://example/foo 
  14.   controller == foo 
  15. Module + controller: 
  16. http://example/blog/archive 
  17.   module   == blog 
  18.   controller == archive 
  19. Module + controller + action: 
  20. http://example/blog/archive/list 
  21.   module   == blog 
  22.   controller == archive 
  23.   action   == list 
  24. Module + controller + action + params: 
  25. http://example/blog/archive/list/sort/alpha/date/desc 
  26.   module   == blog 
  27.   controller == archive 
  28.   action   == list 
  29.   sort    == alpha 
  30.   date    == desc 

缺省路由是存储在RewriteRouter名(index)为'default'的简单的Zend_Controller_Router_Route_Module对象。它被创建多多少少象下面这样:

  1. $compat = new Zend_Controller_Router_Route_Module(array(), 
  2.                          $dispatcher
  3.                          $request); 
  4. $this->addRoute('default'$compat); 

如果你不想这个特别的缺省路由在你的路由计划中,你可以重写你自己的‘缺省'路由(例如,把它存储在'default'名下)或用removeDefaultRoutes()完全清除它:

// Remove any default routes

$router->removeDefaultRoutes();

为了增加路由的灵活性,方便自定义新的路由类型,Zend_Controller_Router定义了Zend_Controller_Router_Route_Interface接口和类Zend_Controller_Router_Route_Abstract,实现相应的类方法即可定义路由类型,为开发提供了便利。

Zend_Controller_Router的路由类型

Zend_Controller_Router默认提供了以下路由类型,分别为:

Zend_Controller_Router_Route

Zend_Controller_Router_Route_Static

Zend_Controller_Router_Route_Regex

Zend_Controller_Router_Route_Hostname

Zend_Controller_Router_Route_Module

Zend_Controller_Router_Route_Chain

Zend_Controller_Router_Route

Zend_Controller_Router_Route是标准的框架路由。它结合了灵活路由定义的易用性。每个路由包含了基本的URL映射(静态的和动态的部分(变量))并且可以被缺省地初始化,也可以根据不同的要求初始化。

让我们想象一下我们假设的应用程序将需要一些广域内容作者的信息页面。我们想能够把浏览器指向http://domain.com/author/martel去看一个叫"martel"的信息。有这样功能的路由看起来是这样的:

  1. $route = new Zend_Controller_Router_Route( 
  2.   'author/:username'
  3.   array
  4.     'controller' => 'profile'
  5.     'action'   => 'userinfo' 
  6.   ) 
  7. ); 
  8. $router->addRoute('user'$route); 

在Zend_Controller_Router_Route里构造函数的第一个参数是路由的定义,它将匹配一个URL,路由定义包含静态的和动态部分,它们由正斜杠('/')符分开,静态部分只是简单的字符:author。动态部分,被叫做变量,用预设的冒号来标记变量名::username。

Note: 字符的的用法

当前的实现允许你使用任何字符(正斜杠除外)作为变量标识符,但强烈建议只使用PHP使用的变量标识符。将来的实现也许会改变这个行为,它可能会导致在你的代码里有隐藏的bugs。

当你把浏览器指向http://domain.com/author/martel这个例子的路由应该被匹配,它所有的变量将被注入到Zend_Controller_Request对象并在ProfileController可访问。由这个例子返回的变量可能会被表示为如下键和值配对的数组:

  1. $values = array
  2.   'username'  => 'martel'
  3.   'controller' => 'profile'
  4.   'action'   => 'userinfo' 
  5. ); 

稍后,基于这些值,Zend_Controller_Dispatcher_Standard应该调用ProfileController类(在缺省模块中)中的userinfoAction()方法,你将依靠Zend_Controller_Action::_getParam()或者Zend_Controller_Request::getParam()方法能够访问所有的变量:

  1. public function userinfoAction() 
  2.   $request = $this->getRequest(); 
  3.   $username = $request->getParam('username'); 
  4.   $username = $this->_getParam('username'); 

路由定义可以包一个额外的特别字符-通配符-表示为'*'号,它被用来取得参数,和缺省模块路由类似(在URI中定义的var=>value),下面的路由多多少少地模仿了模块路由的行为:

  1. $route = new Zend_Controller_Router_Route( 
  2.   ':module/:controller/:action/*'
  3.   array('module' => 'default'
  4. ); 
  5. $router->addRoute('default'$route); 

变量缺省

在路由中每个变量可以有一个缺省值,这就是Zend_Controller_Router_Route中构造函数使用的第二个变量,这个参数是一个数组,在数组中键表示变量名,值就是期望的缺省值:

  1. $route = new Zend_Controller_Router_Route( 
  2.   'archive/:year'
  3.   array('year' => 2006) 
  4. ); 
  5. $router->addRoute('archive'$route); 

上述路由将匹配象http://domain.com/archive/2005和http://example.com/archive的URLs,对于后者变量year将有一个初始的缺省值为2006。

这个例子将导致注入一个year变量给请求对象,应为没有路由信息出现(没有控制器和动作参数被定义),应用程序将被派遣给缺省的控制器和动作方法(它们都在Zend_Controller_Dispatcher_Abstract被定义),为使它更可用,你必须提供一个有效的控制器和动作作为路由的缺省值:

  1. $route = new Zend_Controller_Router_Route( 
  2.   'archive/:year'
  3.   array
  4.     'year'    => 2006, 
  5.     'controller' => 'archive'
  6.     'action'   => 'show' 
  7.   ) 
  8. ); 
  9. $router->addRoute('archive'$route); 

这个路由将导致派遣给ArchiveController类的showAction()方法。

变量请求

当变量请求被设定,第三个参数可以加给Zend_Controller_Router_Route的构造函数,这些被定义为正则表达式的一部分:

  1. $route = new Zend_Controller_Router_Route( 
  2.   'archive/:year'
  3.   array
  4.     'year'    => 2006, 
  5.     'controller' => 'archive'
  6.     'action'   => 'show' 
  7.   ), 
  8.   array('year' => '\d+'
  9. ); 
  10. $router->addRoute('archive'$route); 

用上述定义的路由,路由器仅当year变量包含数字数据将匹配它,例如http://domain.com/archive/2345。象http://example.com/archive/test这样的URL将不被匹配并且控制将被传递给在链中的下一个路由。

主机名路由

你也可以使用主机名做路由匹配,对简单的匹配使用静态主机名选项:

  1. $route = new Zend_Controller_Router_Route( 
  2.   array
  3.     'host' => 'blog.mysite.com'
  4.     'path' => 'archive' 
  5.   ), 
  6.   array
  7.     'module'   => 'blog'
  8.     'controller' => 'archive'
  9.     'action'   => 'index' 
  10.   ) 
  11. ); 
  12. $router->addRoute('archive'$route); 

如果你想匹配参数在主机名里,使用 regex 选项。在下面例子中,子域为动作控制器被用作用户名参数。 当组装路由时,你可以给出用户名为参数,就像你用其它路径参数一样:

  1. $route = new Zend_Controller_Router_Route( 
  2.   array
  3.     'host' => array
  4.       'regex'  => '([a-z]+).mysite.com'
  5.       'reverse' => '%s.mysite.com' 
  6.       'params' => array
  7.         1 => 'username' 
  8.       ) 
  9.     ), 
  10.     'path' => '' 
  11.   ), 
  12.   array
  13.     'module'   => 'users'
  14.     'controller' => 'profile'
  15.     'action'   => 'index' 
  16.   ) 
  17. ); 
  18. $router->addRoute('profile'$route); 
  19. Zend_Controller_Router_Route_Static 

设置固定不变的路由:

  1. $route = new Zend_Controller_Router_Route_Static( 
  2.   'login'
  3.   array('controller' => 'auth''action' => 'login'
  4. ); 
  5. $router->addRoute('login'$route); 

上面的路由将匹配http://domain.com/login的URL,并分派到 AuthController::loginAction().

Zend_Controller_Router_Route_Regex

除了缺省的和静态的路由类型外,正则表达式路由类型也可用。这个路由比其它路由更强更灵活,只是稍微有点复杂。同时,它应该比标准路由快。

象标准路由一样,这个路由必须用路由定义和一些缺省条件来初始化。让我们创建一个archive路由作为例子,和先前定义的类似,这次只是用了Regex:

  1. $route = new Zend_Controller_Router_Route_Regex( 
  2.   'archive/(\d+)'
  3.   array
  4.     'controller' => 'archive'
  5.     'action'   => 'show' 
  6.   ) 
  7. ); 
  8. $router->addRoute('archive'$route); 

每个定义的regex子模式将被注入到请求对象里,同上述的例子,再成功匹配http://domain.com/archive/2006之后,结果值的数组看起来象这样:

  1. $values = array
  2.   1      => '2006'
  3.   'controller' => 'archive'
  4.   'action'   => 'show' 
  5. ); 

Note: 在匹配之前,开头和结尾的斜杠从路由器里的URL中去除掉了,结果,匹配http://domain.com/foo/bar/,需要foo/bar这样的regex,而不是/foo/bar。

Note: 行开头和行结尾符号(分别为'^' 和 '$')被自动预先追加到所有表达式,这样,你不需要在你的正则表达式里用它们,你应该匹配整个字符串。

Note: 这个路由类使用#符作为分隔符。这意味着你将需要避免哈希符('#')但不是正斜杠('/')在你的路由定义里。因为'#'符(名称为锚)很少被传给webserver,你将几乎不需要在你的regex里使用它。

你可以用通常的办法获得已定义的子模式的内容:

  1. public function showAction() 
  2.   $request = $this->getRequest(); 
  3.   $year  = $request->getParam(1); // $year = '2006'; 

Note: 注意这个键是整数(1) 而不是字符串('1')。

因为'year'的缺省没有设置,这个路由将和它的标准路由副本不是非常精确地相同。即使我们为'year'声明一个缺省并使子模式可选,也不清楚是否会在拖尾斜杠(trailing slash)上还将有问题。方案是使整个'year'部分和斜杠一起可选但只抓取数字部分:(这段比较绕口,请校对者仔细看看,谢谢 Jason Qi)

  1. $route = new Zend_Controller_Router_Route_Regex( 
  2.   'archive(?:/(\d+))?'
  3.   array
  4.     1      => '2006'
  5.     'controller' => 'archive'
  6.     'action'   => 'show' 
  7.   ) 
  8. ); 
  9. $router->addRoute('archive'$route); 

让我们看看你可能注意到的问题,给参数使用基于整数的键不是容易管理的办法,今后可能会有问题,这就是为什么有第三个参数,这是个联合数组表示一个regex子模式到参数名键的映射,我们来看看一个简单的例子:

  1. $route = new Zend_Controller_Router_Route_Regex( 
  2.   'archive/(\d+)'
  3.   array
  4.     'controller' => 'archive'
  5.     'action' => 'show' 
  6.   ), 
  7.   array
  8.     1 => 'year' 
  9.   ) 
  10. ); 
  11. $router->addRoute('archive'$route); 

这将导致下面的值被注入到请求:

  1. $values = array
  2.   'year'    => '2006'
  3.   'controller' => 'archive'
  4.   'action'   => 'show' 
  5. ); 

这个映射被任何目录来定义使它能工作于任何环境,键可以包含变量名或子模式索引:

  1. $route = new Zend_Controller_Router_Route_Regex( 
  2.   'archive/(\d+)'
  3.   array( ... ), 
  4.   array(1 => 'year'
  5. ); 
  6. // OR 
  7. $route = new Zend_Controller_Router_Route_Regex( 
  8.   'archive/(\d+)'
  9.   array( ... ), 
  10.   array('year' => 1) 
  11. ); 

Note: 子模式键必须用整数表示。

注意在请求值中的数字索引不见了,代替的是一个名字变量,当然如果你愿意可以把数字和名字变量混合使用:

  1. $route = new Zend_Controller_Router_Route_Regex( 
  2.   'archive/(\d+)/page/(\d+)'
  3.   array( ... ), 
  4.   array('year' => 1) 
  5. ); 

这将导致在请求中有混合的值可用,例如:URLhttp://domain.com/archive/2006/page/10将在下列结果中:

  1. $values = array
  2.   'year'    => '2006'
  3.   2      => 10, 
  4.   'controller' => 'archive'
  5.   'action'   => 'show' 
  6. ); 

因为regex模型不容易颠倒,如果你想用URL助手或这个类中的 assemble方法,你需要准备一个颠倒的URL,这个颠倒的路径用可由sprintf()解析的字符串来表示并定义为第四个构造参数:

  1. $route = new Zend_Controller_Router_Route_Regex( 
  2.   'archive/(\d+)'
  3.   array( ... ), 
  4.   array('year' => 1), 
  5.   'archive/%s' 
  6. ); 

所有这些都已经可能由标准路由对象完成,那么使用Regex路由的好处在哪里?首先,它允许你不受限制地描述任何类型的URL。想象一下你有一个博客并希望创建象http://domain.com/blog/archive/01-Using_the_Regex_Router.html这样的URLs,还有把解析它路径元素中的最后部分,01-Using_the_Regex_Router.html,到一个文章的ID和文章的标题/描述;这不可能由标准路由完成。用Regex路由,你可以做象下面的方案:

  1. $route = new Zend_Controller_Router_Route_Regex( 
  2.   'blog/archive/(\d+)-(.+)\.html'
  3.   array
  4.     'controller' => 'blog'
  5.     'action'   => 'view' 
  6.   ), 
  7.   array
  8.     1 => 'id'
  9.     2 => 'description' 
  10.   ), 
  11.   'blog/archive/%d-%s.html' 
  12. ); 
  13. $router->addRoute('blogArchive'$route); 

正如你所看到的,这个在标准路由上添加了巨大的灵活性。

通过配置文件定义路由规则

例如:

  1. [production] 
  2. routes.archive.route = "archive/:year/*" 
  3. routes.archive.defaults.controller = archive 
  4. routes.archive.defaults.action = show 
  5. routes.archive.defaults.year = 2000 
  6. routes.archive.reqs.year = "\d+" 
  7. routes.news.type = "Zend_Controller_Router_Route_Static" 
  8. routes.news.route = "news" 
  9. routes.news.defaults.controller = "news" 
  10. routes.news.defaults.action = "list" 
  11. routes.archive.type = "Zend_Controller_Router_Route_Regex" 
  12. routes.archive.route = "archive/(\d+)" 
  13. routes.archive.defaults.controller = "archive" 
  14. routes.archive.defaults.action = "show" 
  15. routes.archive.map.1 = "year" 
  16. ; OR: routes.archive.map.year = 1 

上述的INI文件可以被读进Zend_Config对象:

$config = new Zend_Config_Ini('/path/to/config.ini', 'production');

$router = new Zend_Controller_Router_Rewrite();

$router->addConfig($config, 'routes');

在上面的例子中,我们告诉路由器去使用INI文件'routes'一节给它的路由。每个在这个节下的顶级键将用来定义路由名;上述例子定义了路由'archive'和'news'。每个路由接着要求,至少,一个'route'条目和一个或更多'defaults'条目;可选地,一个或更多'reqs'('required'的简写)可能要求提供。总之,这些相对应的三个参数提供给Zend_Controller_Router_Route_Interface对象。一个选项键,'type',可用来指定路由类的类型给特殊的路由;缺省地,它使用Zend_Controller_Router_Route。在上述例子中,'news'路由被定义来使用Zend_Controller_Router_Route_Static。

自定义路由类型

标准的rewrite路由器应当最大限度提供你所需的功能;大多时候,为了通过已知的路由提供新的或修改的功能,你将只需要创建一个新的路由类型

那就是说,你可能想要用不同的路由范例,接口Zend_Controller_Router_Interface提供了需要最少的信息来创建路由器,并包含一个单个的方法。

  1. interface Zend_Controller_Router_Interface 
  2.  /** 
  3.   * @param Zend_Controller_Request_Abstract $request 
  4.   * @throws Zend_Controller_Router_Exception 
  5.   * @return Zend_Controller_Request_Abstract 
  6.   */ 
  7.  public function route(Zend_Controller_Request_Abstract $request); 

路由只发生一次:当请求第一次接收到系统,路由器的意图是基于请求的环境决定控制器、动作和可选的参数,并把它们发给请求,请求对象接着传递给派遣器,如果不可能映射一个路由到一个派遣令牌,路由器对请求对象就什么也不做。

Tags: Zend_Controller_Router

分享到: