2
0

Zend_Controller-Router-Route-Regex.xml 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <!-- EN-Revision: 24249 -->
  3. <!-- Reviewed: no -->
  4. <sect3 id="zend.controller.router.routes.regex">
  5. <title>Zend_Controller_Router_Route_Regex</title>
  6. <para>
  7. En plus des routes par défaut, et statique, les routes exprimées par expression
  8. régulière sont acceptées. Ces routes sont plus puissantes que les autres, mais aussi plus
  9. complexes à mettre en oeuvre et un peu plus lentes en matière d'analyse.
  10. </para>
  11. <para>
  12. Comme les routes standards, cette route doit être initialisée avec une définition et
  13. des valeurs par défaut. Créons par exemple avec une route "archive" en utilisant les routes
  14. par expressions régulières&#160;:
  15. </para>
  16. <programlisting language="php"><![CDATA[
  17. $route = new Zend_Controller_Router_Route_Regex(
  18. 'archive/(\d+)',
  19. array(
  20. 'controller' => 'archive',
  21. 'action' => 'show'
  22. )
  23. );
  24. $router->addRoute('archive', $route);
  25. ]]></programlisting>
  26. <para>
  27. Chaque motif d'expression régulière sera injecté dans l'objet de requête. Avec
  28. l'exemple ci-dessus, en utilisant <code>http://domain.com/archive/2006</code>, la tableau
  29. résultat devrait ressembler à&#160;:
  30. </para>
  31. <programlisting language="php"><![CDATA[
  32. $values = array(
  33. 1 => '2006',
  34. 'controller' => 'archive',
  35. 'action' => 'show'
  36. );
  37. ]]></programlisting>
  38. <note>
  39. <para>
  40. Les slashs de début et de fin sont supprimés de l'URL dans le routeur (avant
  41. l'intervention de la route).Ainsi, pour faire correspondre l'URL
  42. <code>http://domain.com/foo/bar/</code>, il faudra une expression du style
  43. <code>foo/bar</code>, et non pas <code>/foo/bar</code>.
  44. </para>
  45. </note>
  46. <note>
  47. <para>
  48. Les caractères de spécification de début et fin d'expression sont automatiquement
  49. rajoutés au motif ('^' et '$', respectivement). De ce fait, vous ne devriez pas les
  50. utiliser manuellement.
  51. </para>
  52. </note>
  53. <note>
  54. <para>
  55. Cette classe de route utilise le séparateur <code>#</code> comme délimiteur de
  56. motif. Vous devrez donc échapper ce caractère si vous l'utilisez, et non pas le slash
  57. (par défaut pour un motif d'expression régulière). Le caractère "#" est cependant
  58. rarement utilisé dans une <acronym>URL</acronym>.
  59. </para>
  60. </note>
  61. <para>Vous pouvez retourner le contenu des sous-masques&#160;:</para>
  62. <programlisting language="php"><![CDATA[
  63. public function showAction()
  64. {
  65. $request = $this->getRequest();
  66. $year = $request->getParam(1); // $year = '2006';
  67. }
  68. ]]></programlisting>
  69. <note>
  70. <para>Attention, la clé est un entier (1), et non une chaîne ('1').</para>
  71. </note>
  72. <para>
  73. Cette route ne fonctionnera pas encore tout à fait comme la route standard, car la
  74. valeur par défaut pour "<code>year</code>" n'est pas indiquée. Attention par contre, vous
  75. risquez d'avoir un problème avec les slashs finaux même si nous déclarons une valeur par
  76. défaut pour "<code>year</code>" et que celui-ci est facultatif. La solution consiste à
  77. traiter ce slash, mais sans le capturer&#160;:
  78. </para>
  79. <programlisting language="php"><![CDATA[
  80. $route = new Zend_Controller_Router_Route_Regex(
  81. 'archive(?:/(\d+))?',
  82. array(
  83. 1 => '2006',
  84. 'controller' => 'archive',
  85. 'action' => 'show'
  86. )
  87. );
  88. $router->addRoute('archive', $route);
  89. ]]></programlisting>
  90. <para>
  91. Nous voyons apparaître tout de même un problème : gérer des chiffres, comme clés pour
  92. les paramètres n'est pas très intuitif. C'est là qu'entre en jeu le troisième paramètre du
  93. constructeur de <classname>Zend_Controller_Router_Route_Regex</classname>. Il accepte un
  94. tableau faisant correspondre les numéros des paramètres et leur nom respectif&#160;:
  95. </para>
  96. <programlisting language="php"><![CDATA[
  97. $route = new Zend_Controller_Router_Route_Regex(
  98. 'archive/(\d+)',
  99. array(
  100. 'controller' => 'archive',
  101. 'action' => 'show'
  102. ),
  103. array(
  104. 1 => 'year'
  105. )
  106. );
  107. $router->addRoute('archive', $route);
  108. ]]></programlisting>
  109. <para>Les valeurs suivantes seront injectées dans l'objet de requête&#160;:</para>
  110. <programlisting language="php"><![CDATA[
  111. $values = array(
  112. 'year' => '2006',
  113. 'controller' => 'archive',
  114. 'action' => 'show'
  115. );
  116. ]]></programlisting>
  117. <para>Il est aussi possible d'inverser les clé et valeurs du tableau&#160;:</para>
  118. <programlisting language="php"><![CDATA[
  119. $route = new Zend_Controller_Router_Route_Regex(
  120. 'archive/(\d+)',
  121. array( ... ),
  122. array(1 => 'year')
  123. );
  124. // OU
  125. $route = new Zend_Controller_Router_Route_Regex(
  126. 'archive/(\d+)',
  127. array( ... ),
  128. array('year' => 1)
  129. );
  130. ]]></programlisting>
  131. <note>
  132. <para>Attention de toujours manipuler des entiers (1 et non "1")</para>
  133. </note>
  134. <para>
  135. Si vous inversez comme dans le deuxième cas de l'exemple ci-dessus, la clé alors
  136. reçue par l'objet de requête ne représente plus un chiffre, mais le nom du paramètre. Vous
  137. pouvez évidemment mixer les comportements&#160;:
  138. </para>
  139. <programlisting language="php"><![CDATA[
  140. $route = new Zend_Controller_Router_Route_Regex(
  141. 'archive/(\d+)/page/(\d+)',
  142. array( ... ),
  143. array('year' => 1)
  144. );
  145. ]]></programlisting>
  146. <para>
  147. Si nous appelons l'URL <code>http://domain.com/archive/2006/page/10</code> avec la
  148. route définie ci-dessus, les paramètres trouvés seront&#160;:
  149. </para>
  150. <programlisting language="php"><![CDATA[
  151. $values = array(
  152. 'year' => '2006',
  153. 2 => 10,
  154. 'controller' => 'archive',
  155. 'action' => 'show'
  156. );
  157. ]]></programlisting>
  158. <para>
  159. Étant donné que les route par expression régulière ne sont pas facilement réversible,
  160. vous devrez préparer le motif vous-même dans le but d'utiliser l'aide de vue "url". Ce
  161. chemin inverse doit être défini comme une chaîne traitable par la fonction
  162. <methodname>sprintf()</methodname> de <acronym>PHP</acronym>, et définie en quatrième paramètre du constructeur de la
  163. route Regex&#160;:
  164. </para>
  165. <programlisting language="php"><![CDATA[
  166. $route = new Zend_Controller_Router_Route_Regex(
  167. 'archive/(\d+)',
  168. array( ... ),
  169. array('year' => 1),
  170. 'archive/%s'
  171. );
  172. ]]></programlisting>
  173. <para>
  174. Quels sont donc les avantages des routes par expressions régulières (Regex)&#160;? C'est
  175. que vous pouvez décrire n'importe quelle <acronym>URL</acronym> avec. Imaginez un blog, vous voulez créer des
  176. <acronym>URL</acronym>s du type <code>http://domain.com/blog/archive/01-Using_the_Regex_Router.html</code>,
  177. afin de décomposer la dernière partie de l'URL en un ID d'article, et un cours texte
  178. descriptif (<code>01-Using_the_Regex_Router.html</code>). Ceci n'est pas possible avec la
  179. route standard. En revanche, avec la route Regex, vous pouvez écrire&#160;:
  180. </para>
  181. <programlisting language="php"><![CDATA[
  182. $route = new Zend_Controller_Router_Route_Regex(
  183. 'blog/archive/(\d+)-(.+)\.html',
  184. array(
  185. 'controller' => 'blog',
  186. 'action' => 'view'
  187. ),
  188. array(
  189. 1 => 'id',
  190. 2 => 'description'
  191. ),
  192. 'blog/archive/%d-%s.html'
  193. );
  194. $router->addRoute('blogArchive', $route);
  195. ]]></programlisting>
  196. <para>
  197. Comme vous le voyez, ce type de route ajoute une solution flexible concernant la
  198. gestion des <acronym>URL</acronym>s et leur routage.
  199. </para>
  200. </sect3>