1 <sect1 id="zend.controller.router" xmlns:xi="http://www.w3.org/2001/XInclude">
3 <sect2 id="zend.controller.router.introduction">
6 <code>Zend_Controller_Router_Rewrite</code>是标准的框架路由器。路由是个过程,在这个过程中它取出URI的端点(跟着基本URL的URI的那部分)并把它分解成参数来决定哪个模块、哪个控制器和控制器中的哪个动作应该接受请求。模块、控制器、动作和其它参数被打包到<code>Zend_Controller_Request_Http</code>对象,接着这个对象由<code>Zend_Controller_Dispatcher_Standard</code>来处理。路由只发生一次:当请求最初被接收和第一个控制器被派遣之前。
10 <code>Zend_Controller_Router_Rewrite</code>被设计来考虑使用纯php结构时mod_rewrite-like的功能性。它非常宽松地基于Ruby on Rails并且不要求任何先前的web服务器URL rewriting的知识。它被设计来和单个Apache的 mod_rewrite规则(其中之一)一起工作:
13 <programlisting role="php"><![CDATA[
15 RewriteRule !\.(js|ico|gif|jpg|png|css)$ index.php
23 <programlisting role="php"><![CDATA[
25 RewriteCond %{SCRIPT_FILENAME} !-f
26 RewriteCond %{SCRIPT_FILENAME} !-d
27 RewriteRule ^(.*)$ index.php/$1
32 如果<ulink url="http://www.isapirewrite.com">Isapi_Rewrite</ulink>已经用下列的rewrite规则被安装为一个Isapi扩展, rewrite路由器也可以和IIS web服务器一起使用:
35 <programlisting role="php"><![CDATA[
36 RewriteRule ^[\w/\%]*(?:\.(?!(?:js|ico|gif|jpg|png|css)$)[\w\%]*$)? /index.php [I]
41 <title>IIS Isapi_Rewrite</title>
43 当使用IIS,<code>$_SERVER['REQUEST_URI']</code>将要么不存在,要么被设置成一个空串。在这个例子中,<code>Zend_Controller_Request_Http</code>将企图使用被Isapi_Rewrite扩展设置的<code>$_SERVER['HTTP_X_REWRITE_URL']</code>的值。
48 如果使用 Lighttpd,下面的 rewrite 规则有效:
51 <programlisting role="lighttpd"><![CDATA[
53 ".*\?(.*)$" => "/index.php?$1",
54 ".*\.(js|ico|gif|jpg|png|css)$" => "$0",
61 <sect2 id="zend.controller.router.usage">
65 为正确使用rewrite路由器你必须初始化它,添加一些用户定义的路由并注入到控制器。下面的代码示例这个过程:
68 <programlisting role="php"><![CDATA[
71 $router = $ctrl->getRouter(); // returns a rewrite router by default
74 new Zend_Controller_Router_Route('user/:username',
75 array('controller' => 'user',
82 <sect2 id="zend.controller.router.basic">
83 <title>基本的Rewrite路由器操作</title>
86 RewriteRouter的核心是用户定义路由的定义。路由通过调用RewriteRouter的addRoute方法和传递一个由类实现的<code>Zend_Controller_Router_Route_Interface</code>的新的实例被添加。例如:
89 <programlisting role="php"><![CDATA[
90 $router->addRoute('user',
91 new Zend_Controller_Router_Route('user/:username'));
96 Rewrite 路由器带有四个基本类型的路由(其中一个是特殊的):
99 <itemizedlist mark="opencircle">
100 <listitem><para><xref linkend="zend.controller.router.routes.standard" /></para></listitem>
101 <listitem><para><xref linkend="zend.controller.router.routes.static" /></para></listitem>
102 <listitem><para><xref linkend="zend.controller.router.routes.regex" /></para></listitem>
103 <listitem><para><xref linkend="zend.controller.router.default-routes" /> *</para></listitem>
107 路由可以被使用无数次来创建链或用户定义的应用程序路由计划。你可以在任何配置中使用任何数量的路由,除了模块路由以外,它最好被用一次并作为通用路由(例如,作为缺省的)。每个路由将在稍后详细描述。
111 addRoute的第一个参数是路由名。它用来作为权柄从路由器中取得路由(例如,for URL generation purposes)。第二个参数是路由自己。
116 路由名最普通的用法是通过Zend_View_url助手的方法:
119 <programlisting role="php"><![CDATA[
120 "<?= $this->url(array('username' => 'martel'), 'user') ?>">Martel</a>
125 它将导致在 href: <code>user/martel</code>.
130 路由是一个简单的过程,这个过程通过所有提供的路由和匹配它的当前请求的URI定义来迭代。当一个正匹配被发现,变量值从路由实例返回并注入到<code>Zend_Controller_Request</code>对象以备将来在派遣器和用户创建的控制器中使用。如果是负匹配,在链中的下个路由被检查。
136 用倒序来匹配路由确保最通用的路由被首先定义。
143 从路由返回的值来自于URL参数或用于定义的缺省值。这些变量以后可通过<code>Zend_Controller_Request::getParam()</code> 或 <code>Zend_Controller_Action::_getParam()</code> 方法来访问。
148 有三个特殊的变量可用于你的路由-'module'、 'controller' 和 'action'。这些特殊的变量被Zend_Controller_Dispatcher用来找出控制器和动作然后派遣过去。
154 如果你选择通过 <code>setControllerKey</code> 和 <code>setActionKey</code>方法的方式来改变缺省值,这些特殊变量的名字可能会不同。
160 <sect2 id="zend.controller.router.default-routes">
164 Zend_Controller_Router_Rewrite 和缺省路由一起预先配置,它将以<code>controller/action</code>的形式匹配URIs。另外,模块名可以被指定作为第一个路径参数,允许这种<code>module/controller/action</code>形式的URIs。最后,它也将缺省地匹配任何另外的追加到URI的参数-<code>controller/action/var1/value1/var2/value2</code>。
171 <programlisting role="php"><![CDATA[
172 // Assuming the following:
173 $ctrl->setControllerDirectory(
175 'default' => '/path/to/default/controllers',
176 'news' => '/path/to/news/controllers',
177 'blog' => '/path/to/blog/controllers'
185 Invalid module maps to controller name:
190 http://example/blog/archive
192 controller == archive
194 Module + controller + action:
195 http://example/blog/archive/list
197 controller == archive
200 Module + controller + action + params:
201 http://example/blog/archive/list/sort/alpha/date/desc
203 controller == archive
211 缺省路由是存储在RewriteRouter名(index)为'default'的简单的<code>Zend_Controller_Router_Route_Module</code>对象。它被创建多多少少象下面这样:
214 <programlisting role="php"><![CDATA[
215 $compat = new Zend_Controller_Router_Route_Module(array(),
218 $this->addRoute('default', $compat);
223 如果你不想这个特别的缺省路由在你的路由计划中,你可以重写你自己的‘缺省’路由(例如,把它存储在'default'名下)或用<code>removeDefaultRoutes()</code>完全清除它:
226 <programlisting role="php"><![CDATA[
227 // Remove any default routes
228 $router->removeDefaultRoutes();
234 <sect2 id="zend.controller.router.rewritebase">
235 <title>基本 URL 和子目录</title>
238 rewrite路由器可以被用在子目录(例如,<code>http://domain.com/~user/application-root/</code>),在此例中,应用程序 (<code>/~user/application-root</code>)的基本URL应该能自动被<code>Zend_Controller_Request_Http</code>检测到并使用。
242 如果基本URL被误删除,你可以通过<code>Zend_Controller_Request_Http</code> 和调用 <code>setBaseUrl()</code> 方法(参见<xref linkend="zend.controller.request.http.baseurl" />)用你自己的基本路径重写它。
245 <programlisting role="php"><![CDATA[
246 $request->setBaseUrl('/~user/application-root/');
252 <sect2 id="zend.controller.router.routes">
253 <title>Route Types</title>
254 <xi:include href="Zend_Controller-Router-Route.xml" />
255 <xi:include href="Zend_Controller-Router-Route-Static.xml" />
256 <xi:include href="Zend_Controller-Router-Route-Regex.xml" />
259 <sect2 id="zend.controller.router.add-config">
260 <title>使用 Zend_Config with the RewriteRouter</title>
263 有时候,用新路由更新配置文件比修改代码更方便。这个可能通过<code>addConfig()</code>方法来做。基本上,你创建一个Zend_Config-compatible配置,并在你的代码中读入然后传递给RewriteRouter。
270 <programlisting role="php"><![CDATA[
272 routes.archive.route = "archive/:year/*"
273 routes.archive.defaults.controller = archive
274 routes.archive.defaults.action = show
275 routes.archive.defaults.year = 2000
276 routes.archive.reqs.year = "\d+"
278 routes.news.type = "Zend_Controller_Router_Route_Static"
279 routes.news.route = "news"
280 routes.news.defaults.controller = "news"
281 routes.news.defaults.action = "list"
283 routes.archive.type = "Zend_Controller_Router_Route_Regex"
284 routes.archive.route = "archive/(\d+)"
285 routes.archive.defaults.controller = "archive"
286 routes.archive.defaults.action = "show"
287 routes.archive.map.1 = "year"
288 ; OR: routes.archive.map.year = 1
293 上述的INI文件可以被读进<code>Zend_Config</code>对象:
296 <programlisting role="php"><![CDATA[
297 $config = new Zend_Config_Ini('/path/to/config.ini', 'production');
298 $router = new Zend_Controller_Router_Rewrite();
299 $router->addConfig($config, 'routes');
304 在上面的例子中,我们告诉路由器去使用INI文件'routes'一节给它的路由。每个在这个节下的顶级键将用来定义路由名;上述例子定义了路由'archive'和'news'。每个路由接着要求,至少,一个'route'条目和一个或更多'defaults'条目;可选地,一个或更多'reqs'('required'的简写)可能要求提供。总之,这些相对应的三个参数提供给<code>Zend_Controller_Router_Route_Interface</code>对象。一个选项键,'type',可用来指定路由类的类型给特殊的路由;缺省地,它使用<code>Zend_Controller_Router_Route</code>。在上述例子中,'news'路由被定义来使用<code>Zend_Controller_Router_Route_Static</code>。
308 <sect2 id="zend.controller.router.subclassing">
309 <title>Subclassing the Router</title>
312 标准的rewrite路由器应当最大限度提供你所需的功能;大多时候,为了通过已知的路由提供新的或修改的功能,你将只需要创建一个新的路由类型
316 那就是说,你可能想要用不同的路由范例。接口<code>Zend_Controller_Router_Interface</code>提供了需要最少的信息来创建路由器,并包含一个单个的方法。
319 <programlisting role="php"><![CDATA[
320 interface Zend_Controller_Router_Interface
323 * @param Zend_Controller_Request_Abstract $request
324 * @throws Zend_Controller_Router_Exception
325 * @return Zend_Controller_Request_Abstract
327 public function route(Zend_Controller_Request_Abstract $request);
333 路由只发生一次:当请求第一次接收到系统。路由器的意图是基于请求的环境决定控制器、动作和可选的参数,并把它们发给请求。请求对象接着传递给派遣器。如果不可能映射一个路由到一个派遣令牌,路由器对请求对象就什么也不做。