Zend_Db_Adapter.xml 80 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675
  1. <!-- EN-Revision: 13946 -->
  2. <sect1 id="zend.db.adapter">
  3. <title>Zend_Db_Adapter</title>
  4. <para><classname>Zend_Db</classname> et ses autres sous classes proposent une interface de connexion aux bases de données avec
  5. Zend Framework. <classname>Zend_Db_Adapter</classname> est la classe de base que vous utilisez pour vous connecter aux bases
  6. de données (SGBDs). Il y a différentes classes d'adaptateur par SGBD.</para>
  7. <para>Les classes <code>Adapters</code> de <classname>Zend_Db</classname> créent un pont entre les extensions PHP et une
  8. interface commune. Ceci vous aide à écrire des applications déployables avec de multiples SGBDs, demandant peu
  9. d'efforts.</para>
  10. <para>L'interface de la classe d'adaptateur est semblable à celle de l'extension <ulink
  11. url="http://www.php.net/pdo">PHP Data Objects</ulink>. <classname>Zend_Db</classname> propose des classes d'adaptateurs vers
  12. les drivers PDO pour les SGBDs suivants :</para>
  13. <itemizedlist>
  14. <listitem>
  15. <para>IBM DB2 et Informix Dynamic Server (IDS), en utilisant l'extension PHP <ulink
  16. url="http://www.php.net/pdo-ibm">pdo_ibm</ulink>.</para>
  17. </listitem>
  18. <listitem>
  19. <para>MySQL, utilisant l'extension PHP <ulink url="http://www.php.net/pdo-mysql">pdo_mysql</ulink>.</para>
  20. </listitem>
  21. <listitem>
  22. <para>Microsoft SQL Server, utilisant l'extension PHP <ulink
  23. url="http://www.php.net/pdo-mssql">pdo_mssql</ulink>.</para>
  24. </listitem>
  25. <listitem>
  26. <para>Oracle, utilisant l'extension PHP <ulink url="http://www.php.net/pdo-oci">pdo_oci</ulink>.</para>
  27. </listitem>
  28. <listitem>
  29. <para>PostgreSQL, grâce à l'extension PHP <ulink
  30. url="http://www.php.net/pdo-pgsql">pdo_pgsql</ulink>.</para>
  31. </listitem>
  32. <listitem>
  33. <para>SQLite, avec l'extension PHP <ulink url="http://www.php.net/pdo-sqlite">pdo_sqlite</ulink>.</para>
  34. </listitem>
  35. </itemizedlist>
  36. <para>De plus, <classname>Zend_Db</classname> fournit aussi des classes se connectant avec les extensions PHP propres aux
  37. SGBDs (hors PDO donc), pour les SGBDs suivants :</para>
  38. <itemizedlist>
  39. <listitem>
  40. <para>MySQL, utilisant l'extension PHP <ulink url="http://www.php.net/mysqli">mysqli</ulink>.</para>
  41. </listitem>
  42. <listitem>
  43. <para>Oracle, utilisant l'extension PHP <ulink url="http://www.php.net/oci8">oci8</ulink>.</para>
  44. </listitem>
  45. <listitem>
  46. <para>IBM DB2, utilisant l'extension PHP <ulink url="http://www.php.net/ibm_db2">ibm_db2</ulink>.</para>
  47. </listitem>
  48. <listitem>
  49. <para>Firebird/Interbase, utilisant l'extension PHP <ulink
  50. url="http://www.php.net/ibase">php_interbase</ulink></para>
  51. </listitem>
  52. </itemizedlist>
  53. <note>
  54. <para>Chaque <classname>Zend_Db_Adapter</classname> utilise une extension PHP. Vous devez donc les avoir activées pour
  55. utiliser les classes en question. Par exemple, si vous voulez utiliser une classe <classname>Zend_Db_Adapter</classname>
  56. basée sur PDO, vous devrez alors avoir l'extension PDO d'installée, ainsi que l'extension représentant le driver
  57. spécifique à votre SGBD.</para>
  58. </note>
  59. <sect2 id="zend.db.adapter.connecting">
  60. <title>Se connecter à un SGBD en utilisant un adaptateur</title>
  61. <para>Cette section décrit comment créer une instance d'un adaptateur <classname>Zend_Db</classname> de base de
  62. données.</para>
  63. <sect3 id="zend.db.adapter.connecting.constructor">
  64. <title>Utilisation du constructeur du Zend_Db Adapter</title>
  65. <para>Vous pouvez créer une instance d'un adaptateur en utilisant son constructeur. Celui-ci accepte un
  66. paramètre représentant un tableau d'options.</para>
  67. <example id="zend.db.adapter.connecting.constructor.example">
  68. <title>Utiliser le constructeur de l'adaptateur</title>
  69. <programlisting role="php"><![CDATA[
  70. $db = new Zend_Db_Adapter_Pdo_Mysql(array(
  71. 'host' => '127.0.0.1',
  72. 'username' => 'webuser',
  73. 'password' => 'xxxxxxxx',
  74. 'dbname' => 'test'
  75. ));
  76. ]]></programlisting>
  77. </example>
  78. </sect3>
  79. <sect3 id="zend.db.adapter.connecting.factory">
  80. <title>Utiliser la fabrique (Factory) de <classname>Zend_Db</classname></title>
  81. <para>Alternativement, il est possible d'utiliser la méthode statique <classname>Zend_Db::factory()</classname>.
  82. Celle-ci charge dynamiquement la classe d'adaptateur correspondant en utilisant <link
  83. linkend="zend.loader.load.class">Zend_Loader::loadClass()</link>.</para>
  84. <para>Le premier argument est une chaîne désignant l'adaptateur souhaité. Par exemple,
  85. "<code>Pdo_Mysql</code>" va correspondre à la classe <classname>Zend_Db_Adapter_Pdo_Mysql</classname>. Le second
  86. paramètre est un tableau d'options. C'est le même que celui que vous auriez passé au constructeur de la
  87. classe directement.</para>
  88. <example id="zend.db.adapter.connecting.factory.example">
  89. <title>Utilisation de la méthode statique de fabrique de <classname>Zend_Db</classname></title>
  90. <programlisting role="php"><![CDATA[
  91. // Nous n'avons pas besoin de la ligne suivante car Zend_Db_Adapter_Pdo_Mysql
  92. // sera automatiquement chargé par la fabrique Zend_Db.
  93. // require_once 'Zend/Db/Adapter/Pdo/Mysql.php';
  94. // Charge automatiquement la classe Zend_Db_Adapter_Pdo_Mysql
  95. // et en créer une instance.
  96. $db = Zend_Db::factory('Pdo_Mysql', array(
  97. 'host' => '127.0.0.1',
  98. 'username' => 'webuser',
  99. 'password' => 'xxxxxxxx',
  100. 'dbname' => 'test'
  101. ));
  102. ]]></programlisting>
  103. </example>
  104. <para>Si vous créez votre propre classe d'adaptateur qui étend <classname>Zend_Db_Adapter_Abstract</classname> et que
  105. celle-ci ne respecte pas la syntaxe du préfixe package "<classname>Zend_Db_Adapter</classname>", utilisez alors la clé
  106. "<code>adapterNamespace</code>" dans le tableau de configuration passé à la méthode <code>factory()</code>
  107. afin de charger votre adaptateur.</para>
  108. <example id="zend.db.adapter.connecting.factory.example2">
  109. <title>Utilisation de la fabrique avec une classe personnalisée</title>
  110. <programlisting role="php"><![CDATA[
  111. // Charge automatiquement la classe MyProject_Db_Adapter_Pdo_Mysql
  112. // et l'instantie.
  113. $db = Zend_Db::factory('Pdo_Mysql', array(
  114. 'host' => '127.0.0.1',
  115. 'username' => 'webuser',
  116. 'password' => 'xxxxxxxx',
  117. 'dbname' => 'test',
  118. 'adapterNamespace' => 'MyProject_Db_Adapter'
  119. ));
  120. ]]></programlisting>
  121. </example>
  122. </sect3>
  123. <sect3 id="zend.db.adapter.connecting.factory-config">
  124. <title>Utiliser <classname>Zend_Config</classname> avec la fabrique <classname>Zend_Db</classname></title>
  125. <para>Optionnellement, vous pouvez passer un objet de type <link linkend="zend.config">Zend_Config</link> en
  126. tant qu'argument de la méthode <code>factory()</code>, concernant la configuration.</para>
  127. <para>Il est alors nécessaire que l'objet de configuration contienne une propriété <code>adapter</code>, qui
  128. représente une chaîne de caractères décrivant l'adaptateur à utiliser. De plus, l'objet peut aussi contenir
  129. une propriété nommée <code>params</code>, avec toutes les sous propriétés requises pour la configuration de
  130. l'adaptateur.</para>
  131. <example id="zend.db.adapter.connecting.factory.example1">
  132. <title>Utilisation de la fabrique avec un objet de type <classname>Zend_Config</classname></title>
  133. <para>Dans l'exemple qui va suivre, l'objet <classname>Zend_Config</classname> est crée à partir d'un tableau. Il
  134. eut été possible de le créer à partir de fichiers externes, grâce à <link
  135. linkend="zend.config.adapters.ini">Zend_Config_Ini</link> ou <link
  136. linkend="zend.config.adapters.xml">Zend_Config_Xml</link>.</para>
  137. <programlisting role="php"><![CDATA[
  138. $config = new Zend_Config(
  139. array(
  140. 'database' => array(
  141. 'adapter' => 'Mysqli',
  142. 'params' => array(
  143. 'host' => '127.0.0.1',
  144. 'dbname' => 'test',
  145. 'username' => 'webuser',
  146. 'password' => 'secret',
  147. )
  148. )
  149. )
  150. );
  151. $db = Zend_Db::factory($config->database);
  152. ]]></programlisting>
  153. </example>
  154. <para>Le second paramètre de la méthode <code>factory()</code> doit être un tableau associatif décrivant les
  155. paramètres de l'adaptateur à utiliser. Cet argument est optionnel, si un objet de type
  156. <classname>Zend_Config</classname> est utilisé en premier paramètre, alors il est supposé contenir les paramètres, et
  157. le second paramètre de <code>factory()</code> est alors ignoré.</para>
  158. </sect3>
  159. <sect3 id="zend.db.adapter.connecting.parameters">
  160. <title>Paramètres de l'adaptateur (Adapter)</title>
  161. <para>La liste ci dessous explique les différents paramètres acceptés par les classes d'adaptateur
  162. <classname>Zend_Db</classname>.</para>
  163. <itemizedlist>
  164. <listitem>
  165. <para><emphasis role="strong">host</emphasis> : le nom de l'hôte hébergeant le SGBD. Vous pouvez
  166. aussi spécifier une adresse IP. Si le SGBD se situe sur la même machine que l'application PHP,
  167. "localhost" ou "127.0.0.1" devraient alors être utilisés.</para>
  168. </listitem>
  169. <listitem>
  170. <para><emphasis role="strong">username </emphasis>: nom d'utilisateur du compte de connexion au
  171. SGBD.</para>
  172. </listitem>
  173. <listitem>
  174. <para><emphasis role="strong">password</emphasis> : mot de passe de l'utilisateur du compte de
  175. connexion au SGBD.</para>
  176. </listitem>
  177. <listitem>
  178. <para><emphasis role="strong">dbname</emphasis> : nom de la base de données située dans le
  179. SGBD.</para>
  180. </listitem>
  181. <listitem>
  182. <para><emphasis role="strong">port</emphasis> : Certains SGBDs acceptent que l'on spécifie un port
  183. pour d'y connecter. Indiquez le alors ici.</para>
  184. </listitem>
  185. <listitem>
  186. <para><emphasis role="strong">options</emphasis> : Ce paramètre est un tableau associatif d'options
  187. génériques à toutes les classes <classname>Zend_Db_Adapter</classname>.</para>
  188. </listitem>
  189. <listitem>
  190. <para><emphasis role="strong">driver_options</emphasis> : Ce paramètre est un tableau associatif
  191. d'options spécifiques à une extension de SGBD spécifique. Typiquement, il est possible avec ce
  192. paramètre de passer des options (attributs) au driver PDO.</para>
  193. </listitem>
  194. <listitem>
  195. <para><emphasis role="strong">adapterNamespace</emphasis> : fournit le commencement du nom de la
  196. classe d'adaptateur, à utiliser la place de "<classname>Zend_Db_Adapter</classname>". Utilisez ceci si vous
  197. désirez que <code>factory()</code> charge une classe non Zend.</para>
  198. </listitem>
  199. </itemizedlist>
  200. <example id="zend.db.adapter.connecting.parameters.example1">
  201. <title>Passer l'option de gestion de la casse à la fabrique</title>
  202. <para>Vous pouvez spécifier cette option avec la constante <classname>Zend_Db::CASE_FOLDING</classname>. Ceci
  203. correspond à l'attribut <code>ATTR_CASE</code> dans les drivers PDO et IBM DB2, ce qui ajuste la casse
  204. des clés dans les jeux de résultats. Les valeurs possibles possibles sont
  205. <classname>Zend_Db::CASE_NATURAL</classname> (défaut), <classname>Zend_Db::CASE_UPPER</classname>, et
  206. <classname>Zend_Db::CASE_LOWER</classname>.</para>
  207. <programlisting role="php"><![CDATA[
  208. $options = array(
  209. Zend_Db::CASE_FOLDING => Zend_Db::CASE_UPPER
  210. );
  211. $params = array(
  212. 'host' => '127.0.0.1',
  213. 'username' => 'webuser',
  214. 'password' => 'xxxxxxxx',
  215. 'dbname' => 'test',
  216. 'options' => $options
  217. );
  218. $db = Zend_Db::factory('Db2', $params);
  219. ]]></programlisting>
  220. </example>
  221. <example id="zend.db.adapter.connecting.parameters.example2">
  222. <title>Passer l'option d'auto-échappement à la fabrique</title>
  223. <para>Vous pouvez spécifier cette option avec le paramètre <classname>Zend_Db::AUTO_QUOTE_IDENTIFIERS</classname>.
  224. Si la valeur passée est <code>true</code> (par défaut), alors les identifiants tels que les noms de
  225. tables, de colonnes, ou encore les alias SQL, sont échappés (délimités) dans la syntaxe de la requête
  226. SQL générée par l'objet d'adaptateur. Ceci rend l'utilisation de mots SQL contenant des identifiant
  227. spéciaux plus simple. Dans le cas de <code>false</code>, vous devrez vous-même délimiter ces identifiant
  228. avec la méthode <code>quoteIdentifier()</code>.</para>
  229. <programlisting role="php"><![CDATA[
  230. $options = array(
  231. Zend_Db::AUTO_QUOTE_IDENTIFIERS => false
  232. );
  233. $params = array(
  234. 'host' => '127.0.0.1',
  235. 'username' => 'webuser',
  236. 'password' => 'xxxxxxxx',
  237. 'dbname' => 'test',
  238. 'options' => $options
  239. );
  240. $db = Zend_Db::factory('Pdo_Mysql', $params);
  241. ]]></programlisting>
  242. </example>
  243. <example id="zend.db.adapter.connecting.parameters.example3">
  244. <title>Passer des options de driver PDO à la fabrique</title>
  245. <programlisting role="php"><![CDATA[
  246. $pdoParams = array(
  247. PDO::MYSQL_ATTR_USE_BUFFERED_QUERY => true
  248. );
  249. $params = array(
  250. 'host' => '127.0.0.1',
  251. 'username' => 'webuser',
  252. 'password' => 'xxxxxxxx',
  253. 'dbname' => 'test',
  254. 'driver_options' => $pdoParams
  255. );
  256. $db = Zend_Db::factory('Pdo_Mysql', $params);
  257. echo $db->getConnection()
  258. ->getAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY);
  259. ]]></programlisting>
  260. </example>
  261. <example id="zend.db.adapter.connecting.parameters.example4">
  262. <title>Passer des options de sérialisation à la fabrique</title>
  263. <programlisting role="php"><![CDATA[
  264. $options = array(
  265. Zend_Db::ALLOW_SERIALIZATION => false
  266. );
  267. $params = array(
  268. 'host' => '127.0.0.1',
  269. 'username' => 'webuser',
  270. 'password' => 'xxxxxxxx',
  271. 'dbname' => 'test',
  272. 'options' => $options
  273. );
  274. $db = Zend_Db::factory('Pdo_Mysql', $params);
  275. ]]></programlisting>
  276. </example>
  277. </sect3>
  278. <sect3 id="zend.db.adapter.connecting.getconnection">
  279. <title>Gestion des connexions dites paresseuses</title>
  280. <para>La création d'une instance d'une classe d'adaptateur ne crée pas physiquement une connexion au SGBD.
  281. L'adaptateur sauvegarde les paramètres et se connectera physiquement à la demande, la première fois que vous
  282. aurez besoin d'exécuter une requête. Ceci permet d'assurer que la création de l'instance elle-même est
  283. rapide, et ne coûte rien en performances. Vous pouvez donc créer une instance de l'adaptateur, même si vous
  284. ne savez pas si vous allez l'utiliser. Ainsi, si vos paramètres sont incorrects, il faudra attendre la
  285. tentative de connexion au SGBD pour le vérifier réellement.</para>
  286. <para>Si vous voulez forcer l'adaptateur à se connecter au SGBD, utilisez sa méthode
  287. <code>getConnection()</code>. Elle retournera alors un objet représentant la connexion, en fonction de
  288. l'extension PHP utilisée, ou une exception si la connexion n'a pas été réalisée. Par exemple, si votre
  289. adaptateur utilise PDO, le retour sera un objet PDO. La connexion physique au SGBD est alors
  290. réalisée.</para>
  291. <para>Afin de vérifier si les paramètres de connexion au SGBD sont corrects, surveillez les exceptions
  292. envoyées par la méthode <code>getConnection()</code>.</para>
  293. <para>De plus, un adaptateur peut être sérialisé pour être stocké, par exemple, dans une variable de
  294. session. Ceci peut être utile non seulement pour l'adaptateur lui-même, mais aussi pour les autres objets
  295. qui l'agrègent, comme un objet <classname>Zend_Db_Select</classname>. Par défaut, les adaptateurs sont autorisés à
  296. être sérialisés, si vous ne le voulez pas, vous devez passer l'option
  297. <classname>Zend_Db::ALLOW_SERIALIZATION=false</classname>, regardez l'exemple ci-dessus. Afin de respecter le principe
  298. de connexions paresseuses, l'adaptateur ne se reconnectera pas après la désérialisation. Vous devez appeler
  299. vous-même <code>getConnection()</code>. Vous pouvez permettre à l'adaptateur de se reconnecter
  300. automatiquement en utilisant l'option d'adaptateur
  301. <classname>Zend_Db::AUTO_RECONNECT_ON_UNSERIALIZE=true</classname>.</para>
  302. <example id="zend.db.adapter.connecting.getconnection.example">
  303. <title>Gérer les exceptions de connexion</title>
  304. <programlisting role="php"><![CDATA[
  305. try {
  306. $db = Zend_Db::factory('Pdo_Mysql', $parameters);
  307. $db->getConnection();
  308. } catch (Zend_Db_Adapter_Exception $e) {
  309. // probablement mauvais identifiants,
  310. // ou alors le SGBD n'est pas joignable
  311. } catch (Zend_Exception $e) {
  312. // probablement que factory() n'a pas réussi à charger
  313. // la classe de l'adaptateur demandé
  314. }
  315. ]]></programlisting>
  316. </example>
  317. </sect3>
  318. </sect2>
  319. <sect2 id="zend.db.adapter.example-database">
  320. <title>La base de données d'exemple</title>
  321. <para>Dans cette documentation concernant <classname>Zend_Db</classname>, nous utilisons un exemple simple de tables pour
  322. illustrer nos exemples. Ces tables peuvent servir à stocker des informations sur la gestion des bugs dans une
  323. application. La base de données contient quatre tables :</para>
  324. <itemizedlist>
  325. <listitem>
  326. <para><emphasis role="strong">accounts</emphasis> correspond aux informations sur les utilisateurs qui
  327. gèrent les bugs.</para>
  328. </listitem>
  329. <listitem>
  330. <para><emphasis role="strong">products</emphasis> enregistre les produits pour lesquels des bugs vont
  331. être relevés.</para>
  332. </listitem>
  333. <listitem>
  334. <para><emphasis role="strong">bugs</emphasis> est la table qui contient les bugs, à savoir leur état
  335. actuel, la personne ayant relevé le bug, la personne en charge de le corriger, et la personne chargée de
  336. vérifier le correctif.</para>
  337. </listitem>
  338. <listitem>
  339. <para><emphasis role="strong">bugs_products</emphasis> enregistre les relations entre les bugs, et les
  340. produits. C'est une relation plusieurs à plusieurs car un même bug peut faire partie de plusieurs
  341. produits, et un produit peut évidemment posséder plusieurs bugs.</para>
  342. </listitem>
  343. </itemizedlist>
  344. <para>Le pseudo-code SQL suivant représente les tables de notre base de données d'exemple. Ces tables sont
  345. utilisées aussi pour les tests unitaires automatisés de <classname>Zend_Db</classname>.</para>
  346. <programlisting role="sql"><![CDATA[
  347. CREATE TABLE accounts (
  348. account_name VARCHAR(100) NOT NULL PRIMARY KEY
  349. );
  350. CREATE TABLE products (
  351. product_id INTEGER NOT NULL PRIMARY KEY,
  352. product_name VARCHAR(100)
  353. );
  354. CREATE TABLE bugs (
  355. bug_id INTEGER NOT NULL PRIMARY KEY,
  356. bug_description VARCHAR(100),
  357. bug_status VARCHAR(20),
  358. reported_by VARCHAR(100) REFERENCES accounts(account_name),
  359. assigned_to VARCHAR(100) REFERENCES accounts(account_name),
  360. verified_by VARCHAR(100) REFERENCES accounts(account_name)
  361. );
  362. CREATE TABLE bugs_products (
  363. bug_id INTEGER NOT NULL REFERENCES bugs,
  364. product_id INTEGER NOT NULL REFERENCES products,
  365. PRIMARY KEY (bug_id, product_id)
  366. );
  367. ]]></programlisting>
  368. <para>Notez aussi que la table <code>bugs</code> contient plusieurs référence (clés étrangères) vers la table
  369. <code>accounts</code>. Chacune de ces clés peut référencer un enregistrement différent de la table
  370. <code>accounts</code>, pour un bug donné.</para>
  371. <para>Le diagramme qui suit illustre le modèle physique des données.</para>
  372. <para><inlinegraphic align="center" fileref="figures/zend.db.adapter.example-database.png" format="PNG"
  373. scale="100" valign="middle" width="387" /></para>
  374. </sect2>
  375. <sect2 id="zend.db.adapter.select">
  376. <title>Lecture de résultats de requête</title>
  377. <para>Cette section décrit des méthodes de la classe d'adaptateur permettant l'obtention de résultats suivants
  378. une requête SELECT.</para>
  379. <sect3 id="zend.db.adapter.select.fetchall">
  380. <title>Récupérer tous les résultats</title>
  381. <para>Vous pouvez à la fois exécuter une requête SELECT et récupérer tous ses résultats en une seule
  382. manipulation, grâce à la méthode <code>fetchAll()</code>.</para>
  383. <para>Le premier paramètre de cette méthode est une chaîne représentant la requête SELECT à exécuter. Aussi,
  384. ce premier paramètre peut être un objet <link linkend="zend.db.select">Zend_Db_Select</link>, qui sera alors
  385. converti en une chaîne automatiquement.</para>
  386. <para>Le second paramètre de de <code>fetchAll()</code> est un tableau de substitutions des éventuels jokers
  387. présents dans la syntaxe SQL.</para>
  388. <example id="zend.db.adapter.select.fetchall.example">
  389. <title>Utiliser <code>fetchAll()</code></title>
  390. <programlisting role="php"><![CDATA[
  391. $sql = 'SELECT * FROM bugs WHERE bug_id = ?';
  392. $result = $db->fetchAll($sql, 2);
  393. ]]></programlisting>
  394. </example>
  395. </sect3>
  396. <sect3 id="zend.db.adapter.select.fetch-mode">
  397. <title>Changer le mode de récupération (Fetch Mode)</title>
  398. <para>Par défaut, <code>fetchAll()</code> retourne un tableau d'enregistrements. Chaque enregistrement étant
  399. un tableau associatif dont les clés sont les noms des colonnes SQL désirées, ou leurs alias.</para>
  400. <para>Vous pouvez spécifier un mode de récupération de résultats différent, ceci par la méthode
  401. <code>setFetchMode()</code>. Les modes supportés sont identifiés par des constantes :</para>
  402. <itemizedlist>
  403. <listitem>
  404. <para><emphasis role="strong">Zend_Db::FETCH_ASSOC</emphasis>: Retourne un tableau
  405. d'enregistrements. Chaque enregistrement étant un tableau associatif dont les clés sont les noms des
  406. colonnes SQL désirées, ou leurs alias. Il s'agit du mode par défaut utilisé par les classes
  407. Zend_Db_Adapter.</para>
  408. <para>Notez que si votre résultat comporte plusieurs colonnes avec le même nom, par exemple lors
  409. d'une jointure, il ne peut y avoir qu'un clé avec un nom définit dans le tableau de résultat. Vous
  410. devriez toujours utiliser des alias avec le mode FETCH_ASSOC.</para>
  411. <para>Les noms des clés des tableaux correspondants aux noms des colonnes SQL telles que retournées
  412. par le SGBD, vous pouvez spécifier la casse pour ces noms, grâce à l'option
  413. <classname>Zend_Db::CASE_FOLDING</classname>. Spécifiez ceci lors de l'instanciation de votre adaptateur.
  414. Voyez <xref linkend="zend.db.adapter.connecting.parameters.example1" />.</para>
  415. </listitem>
  416. <listitem>
  417. <para><emphasis role="strong">Zend_Db::FETCH_NUM</emphasis>: Retourne les enregistrements dans un
  418. tableau de tableaux. Les tableaux nichés sont indexés par des entiers correspondants à la position
  419. du champ dans la syntaxe SQL SELECT.</para>
  420. </listitem>
  421. <listitem>
  422. <para><emphasis role="strong">Zend_Db::FETCH_BOTH</emphasis>: Retourne les enregistrements dans un
  423. tableau de tableaux. Les tableaux nichés sont indexés à la fois numériquement et lexicalement. C'est
  424. un mode qui réunit FETCH_ASSOC et FETCH_NUM. Ainsi, vous avez deux fois plus d'enregistrements,
  425. chacun d'entre eux étant doublé.</para>
  426. </listitem>
  427. <listitem>
  428. <para><emphasis role="strong">Zend_Db::FETCH_COLUMN</emphasis>: Retourne les enregistrements dans un
  429. tableau de valeurs. Les valeurs correspondent à une des colonnes utilisées dans la requête SQL
  430. SELECT. Par défaut, il s'agit de la colonne à l'index 0.</para>
  431. </listitem>
  432. <listitem>
  433. <para><emphasis role="strong">Zend_Db::FETCH_OBJ</emphasis>: Retourne les enregistrements dans un
  434. tableau d'objets. La classe de ces objets par défaut est la classe intégrée à PHP :
  435. <code>stdClass</code>. Les colonnes des enregistrements sont représentées par les propriétés
  436. publiques des objets.</para>
  437. </listitem>
  438. </itemizedlist>
  439. <example id="zend.db.adapter.select.fetch-mode.example">
  440. <title>Utiliser <code>setFetchMode()</code></title>
  441. <programlisting role="php"><![CDATA[
  442. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  443. $result = $db->fetchAll('SELECT * FROM bugs WHERE bug_id = ?', 2);
  444. // $result est un tableau d'objets
  445. echo $result[0]->bug_description;
  446. ]]></programlisting>
  447. </example>
  448. </sect3>
  449. <sect3 id="zend.db.adapter.select.fetchassoc">
  450. <title>Récupérer un enregistrement comme tableau associatif</title>
  451. <para>La méthode <code>fetchAssoc()</code> retourne des enregistrements sous forme de tableau de tableaux
  452. associatifs, quelque soit la valeur de "fetch mode".</para>
  453. <example id="zend.db.adapter.select.fetchassoc.example">
  454. <title>Utiliser f<code>etchAssoc()</code></title>
  455. <programlisting role="php"><![CDATA[
  456. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  457. $result = $db->fetchAssoc('SELECT * FROM bugs WHERE bug_id = ?', 2);
  458. // $result est un tableau de tableaux associatifs
  459. echo $result[0]['bug_description'];
  460. ]]></programlisting>
  461. </example>
  462. </sect3>
  463. <sect3 id="zend.db.adapter.select.fetchcol">
  464. <title>Récupérer une seule colonne d'un enregistrement</title>
  465. <para>La méthode <code>fetchCol()</code> retourne les enregistrements dans un tableau de valeurs. Les
  466. valeurs correspondent à une des colonnes utilisées dans la requête SQL SELECT, par défaut : la première.
  467. Toute autre colonne sera ignorée. Si vous avez besoin de retourner une autre colonne, voyez <xref
  468. linkend="zend.db.statement.fetching.fetchcolumn" />. Cette méthode est indépendante de la valeur de "fetch
  469. mode".</para>
  470. <example id="zend.db.adapter.select.fetchcol.example">
  471. <title>Utiliser <code>fetchCol()</code></title>
  472. <programlisting role="php"><![CDATA[
  473. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  474. $sql = 'SELECT bug_description, bug_id FROM bugs WHERE bug_id = ?';
  475. $result = $db->fetchCol($sql, 2);
  476. // Contient bug_description ; bug_id n'est pas retourné
  477. echo $result[0];
  478. ]]></programlisting>
  479. </example>
  480. </sect3>
  481. <sect3 id="zend.db.adapter.select.fetchpairs">
  482. <title>Récupérer des paires Clé-Valeur d'enregistrements</title>
  483. <para>La méthode <code>fetchPairs()</code> retourne un tableau de paires clés/valeurs. La clé est le
  484. résultat de la première colonne sélectionnée dans la requête, la valeur est le résultat de la deuxième
  485. colonne sélectionnée dans la requête. Il est donc inutile de sélectionner plus de deux colonnes avec cette
  486. méthode. De même, vous devez sélectionner exactement deux colonnes avec cette méthode, pas moins. Si des
  487. clés ont des doublons, alors ils seront écrasés.</para>
  488. <para>Vous devriez réfléchir votre requête SELECT de manière à ce que la première colonne sélectionnée,
  489. correspondant à la clé du tableau de résultat, soit unique (une clé primaire par exemple). Cette méthode est
  490. indépendante de "fetch mode" éventuellement précédemment défini.</para>
  491. <example id="zend.db.adapter.select.fetchpairs.example">
  492. <title>Utilisation de <code>fetchPairs()</code></title>
  493. <programlisting role="php"><![CDATA[
  494. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  495. $result = $db->fetchPairs('SELECT bug_id, bug_status FROM bugs');
  496. echo $result[2]; // le bug_status correspondant au bug_id numéro 2
  497. ]]></programlisting>
  498. </example>
  499. </sect3>
  500. <sect3 id="zend.db.adapter.select.fetchrow">
  501. <title>Récupérer un seul enregistrement complet</title>
  502. <para>La méthode <code>fetchRow()</code> retourne un et un seul enregistrement (le premier si plusieurs
  503. correspondent), en fonction de "fetch mode" que vous aurez précédemment défini. Cette méthode ressemble donc
  504. à <code>fetchAll()</code> si ce n'est qu'elle ne retournera jamais plus d'un seul enregistrement. Arrangez
  505. vous donc pour que votre SELECT possède une clause WHERE sur une clé primaire.</para>
  506. <example id="zend.db.adapter.select.fetchrow.example">
  507. <title>Utiliser <code>fetchRow()</code></title>
  508. <programlisting role="php"><![CDATA[
  509. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  510. $result = $db->fetchRow('SELECT * FROM bugs WHERE bug_id = 2');
  511. // Ce résultat sera un objet, car le fetch mode en a décidé ainsi
  512. echo $result->bug_description;
  513. ]]></programlisting>
  514. </example>
  515. </sect3>
  516. <sect3 id="zend.db.adapter.select.fetchone">
  517. <title>Récupérer une colonne d'un enregistrement</title>
  518. <para>La méthode <code>fetchOne()</code> est une combinaison des méthodes <code>fetchRow()</code> et
  519. <code>fetchCol()</code>, ainsi elle ne retourne que la première colonne, du premier enregistrement retourné.
  520. La valeur de retour est donc une chaîne de caractères. Toute requête retournant plusieurs colonnes et/ou
  521. plusieurs résultats est donc inutile avec cette méthode.</para>
  522. <example id="zend.db.adapter.select.fetchone.example">
  523. <title>Utiliser <code>fetchOne()</code></title>
  524. <programlisting role="php"><![CDATA[
  525. $result = $db->fetchOne('SELECT bug_status FROM bugs WHERE bug_id = 2');
  526. // ceci est une chaine
  527. echo $result;
  528. ]]></programlisting>
  529. </example>
  530. </sect3>
  531. </sect2>
  532. <sect2 id="zend.db.adapter.write">
  533. <title>Effectuer des changements dans la base de données</title>
  534. <para>Il est bien entendu possible d'utiliser la classe d'adaptateur pour effectuer des changements dans vos
  535. données. Cette section décrit les manières de procéder.</para>
  536. <sect3 id="zend.db.adapter.write.insert">
  537. <title>Insérer des données</title>
  538. <para>Vous pouvez ajouter de nouveaux enregistrements dans une table, grâce à la méthode
  539. <code>insert()</code>. Son premier paramètre est une chaîne qui représente le nom de la table ciblée, le
  540. second paramètre est un tableau associatif liant les noms des colonnes de la table, aux valeurs
  541. souhaitées.</para>
  542. <example id="zend.db.adapter.write.insert.example">
  543. <title>Insertion dans une table</title>
  544. <programlisting role="php"><![CDATA[
  545. $data = array(
  546. 'created_on' => '2007-03-22',
  547. 'bug_description' => 'Something wrong',
  548. 'bug_status' => 'NEW'
  549. );
  550. $db->insert('bugs', $data);
  551. ]]></programlisting>
  552. </example>
  553. <para>Les colonnes non citées dans le tableau associatif sont laissées telles quelles. Ainsi, si le SGBD
  554. possède une valeur DEFAULT pour les colonnes concernées, celle-ci sera utilisée, autrement, NULL sera
  555. utilisé.</para>
  556. <para>Par défaut, les valeurs insérées avec cette méthode sont automatiquement échappées. Ceci pour des
  557. raisons de sécurité, vous n'avez donc pas besoin de vous occuper de ce point là.</para>
  558. <para>Si vous avez besoin d'écrire de la syntaxe SQL, comme des mots réservés, des noms de fonctions SQL,
  559. vous voulez que ceux-ci ne soient pas échappés, et ne soient pas traités comme de vulgaires chaînes de
  560. caractères, mais plutôt comme des expressions. Pour ceci, vous devriez passer ces valeurs dans votre tableau
  561. de données, en tant qu'objets de type <classname>Zend_Db_Expr</classname> au lieu de chaînes de caractères
  562. banales.</para>
  563. <example id="zend.db.adapter.write.insert.example2">
  564. <title>Insérer des expressions dans une table</title>
  565. <programlisting role="php"><![CDATA[
  566. $data = array(
  567. 'created_on' => new Zend_Db_Expr('CURDATE()'),
  568. 'bug_description' => 'Something wrong',
  569. 'bug_status' => 'NEW'
  570. );
  571. $db->insert('bugs', $data);
  572. ]]></programlisting>
  573. </example>
  574. </sect3>
  575. <sect3 id="zend.db.adapter.write.lastinsertid">
  576. <title>Récupérer une valeur générée</title>
  577. <para>Certains SGBDs supportent les clé primaires auto-incrémentées. Une table qui utilise un tel procédé
  578. génère la valeur de la clé automatiquement lors d'une insertion (INSERT). La valeur de retour de la méthode
  579. <code>insert()</code> <emphasis>n'est pas</emphasis> le dernier ID inséré car la table peut ne pas avoir de
  580. clé auto-incrémentée. La valeur de retour est le nombres d'enregistrements affectés (théoriquement
  581. 1).</para>
  582. <para>Si votre table a été définie avec une clé auto-incrémentée, alors vous pouvez appeler la méthode
  583. <code>lastInsertId()</code> après une opération d'insertion. Cette méthode retourne la valeur
  584. auto-incrémentée, générée dans le cadre de la connexion au SGBD.</para>
  585. <example id="zend.db.adapter.write.lastinsertid.example-1">
  586. <title>Utiliser l<code>astInsertId()</code> pour les clés auto-incrémentées</title>
  587. <programlisting role="php"><![CDATA[
  588. $db->insert('bugs', $data);
  589. // retourne la dernière valeur générée par la clé auto-incrémentée
  590. $id = $db->lastInsertId();
  591. ]]></programlisting>
  592. </example>
  593. <para>Certains SGBD supporte un objet de séquence, qui sert à générer des valeurs uniques qui vont servir
  594. pour les clé primaires. Pour supporter ce procédé, la méthode <code>lastInsertId()</code> accepte deux
  595. paramètres optionnels (chaînes de caractères). Ces paramètres nomment la table et la colonne en supposant
  596. que vous ayez respecté la convention qui définit que la séquence est nommée en utilisant le nom de la table
  597. et des colonnes utilisées, avec le suffixe "_seq". Ces conventions sont celles de PostgreSQL pour les
  598. colonnes de type SERIAL. Par exemple, une table "bugs" avec une clé primaire "bug_id" utilisera une séquence
  599. nommée "bugs_bug_id_seq".</para>
  600. <example id="zend.db.adapter.write.lastinsertid.example-2">
  601. <title>Utiliser <code>lastInsertId()</code> avec une séquence</title>
  602. <programlisting role="php"><![CDATA[
  603. $db->insert('bugs', $data);
  604. // retourne la dernière valeur générée par la séquence 'bugs_bug_id_seq'.
  605. $id = $db->lastInsertId('bugs', 'bug_id');
  606. // ceci retourne la dernière valeur générée par la séquence 'bugs_seq'.
  607. $id = $db->lastInsertId('bugs');
  608. ]]></programlisting>
  609. </example>
  610. <para>Si le nom de votre objet de séquence ne suit pas ces conventions de nommage, utilisez alors
  611. <code>lastSequenceId()</code>. Cette méthode prend un paramètre qui nomme la séquence explicitement.</para>
  612. <example id="zend.db.adapter.write.lastinsertid.example-3">
  613. <title>Utilisation de <code>lastSequenceId()</code></title>
  614. <programlisting role="php"><![CDATA[
  615. $db->insert('bugs', $data);
  616. // retourne la dernière valeur générée par la séquence 'bugs_id_gen'.
  617. $id = $db->lastSequenceId('bugs_id_gen');
  618. ]]></programlisting>
  619. </example>
  620. <para>Pour les SGBDs ne supportant pas les séquences, comme MySQL, Microsoft SQL Server, et SQLite, les
  621. arguments passés à la méthode <code>lastInsertId()</code> sont ignorés. La valeur retournée est la dernière
  622. valeur générée pour la dernière requête INSERT, quelque soit la table concernée (pour cette connexion).
  623. Aussi, pour ces SGBDs, la méthode <code>lastSequenceId()</code> retournera toujours
  624. <code>null</code>.</para>
  625. <note>
  626. <title>Pourquoi ne pas utiliser "SELECT MAX(id) FROM table"?</title>
  627. <para>Quelques fois, cette requête retourne la valeur la plus récente de clé primaire insérée dans la
  628. table en question. Cependant, cette technique n'est pas pertinente dans un environnement où beaucoup de
  629. clients insèrent beaucoup de données dans une même table. Il est donc possible qu'un client insère une
  630. donnée entre le moment où la dernière insertion est effectuée, et l'appel de MAX(id), aboutissant ainsi
  631. à un résultat erroné. Il est très difficile de se rendre compte d'un tel comportement.</para>
  632. <para>Utiliser un mode d'isolation transactionnelle très élevé, comme "repeatable read" peut mitiger
  633. plus ou moins les risques, mais certains SGBDs ne supportent pas ce mode de transactions.</para>
  634. <para>De plus, utiliser une requête du type "MAX(id)+1" pour générer une nouvelle valeur de clé primaire
  635. n'est pas sécurisé non plus, car deux client peuvent se connecter simultanément et créer des effets
  636. indésirables.</para>
  637. <para>Tous les SGBDs fournissent un mécanisme de génération de valeurs uniques, et une méthode pour les
  638. récupérer. Ces mécanismes travaillent en dehors du mode transactionnel, et empêchent ainsi deux clients
  639. de générer la même valeur, ou de "se marcher dessus".</para>
  640. </note>
  641. </sect3>
  642. <sect3 id="zend.db.adapter.write.update">
  643. <title>Mettre à jour des données</title>
  644. <para>Vous pouvez mettre à jour des données dans une table en utilisant la méthode <code>update()</code> de
  645. l'adaptateur. Cette méthode accepte trois arguments : le premier est le nom de la table, le deuxième est un
  646. tableau faisant correspondre les noms des colonnes SQL à leurs valeurs désirées.</para>
  647. <para>Les valeurs dans ce tableau sont traitées comme des chaînes. Voyez <xref
  648. linkend="zend.db.adapter.write.insert" /> pour plus d'informations sur la gestion des expressions SQL dans
  649. ce tableau.</para>
  650. <para>Le troisième argument est une chaîne contenant l'expression SQL utilisée comme critère pour la mise à
  651. jour des données dans la table. Les valeurs et les arguments dans ce paramètre ne sont pas échappés pour
  652. vous. Vous devez donc vous assurer de l'éventuel bon échappement des caractères. Voyez <xref
  653. linkend="zend.db.adapter.quoting" /> pour plus d'informations.</para>
  654. <para>La valeur de retour de cette méthode est le nombre d'enregistrements affectés par l'opération de mise
  655. à jour (UPDATE).</para>
  656. <example id="zend.db.adapter.write.update.example">
  657. <title>Mettre à jour des enregistrements</title>
  658. <programlisting role="php"><![CDATA[
  659. $data = array(
  660. 'updated_on' => '2007-03-23',
  661. 'bug_status' => 'FIXED'
  662. );
  663. $n = $db->update('bugs', $data, 'bug_id = 2');
  664. ]]></programlisting>
  665. </example>
  666. <para>Si vous oubliez le troisième paramètre, alors tous les enregistrements de la table sont mis à jour
  667. avec les valeurs spécifiées dans le tableau de données.</para>
  668. <para>Si vous spécifiez un tableau de chaîne en tant que troisième paramètre, alors ces chaînes sont jointes
  669. entre elles avec une opération <code>AND</code>.</para>
  670. <example id="zend.db.adapter.write.update.example-array">
  671. <title>Mettre à jour des enregistrements avec un tableau de données</title>
  672. <programlisting role="php"><![CDATA[
  673. $data = array(
  674. 'updated_on' => '2007-03-23',
  675. 'bug_status' => 'FIXED'
  676. );
  677. $where[] = "reported_by = 'goofy'";
  678. $where[] = "bug_status = 'OPEN'";
  679. $n = $db->update('bugs', $data, $where);
  680. // la requête SQL executée est:
  681. // UPDATE "bugs" SET "update_on" = '2007-03-23', "bug_status" = 'FIXED'
  682. // WHERE ("reported_by" = 'goofy') AND ("bug_status" = 'OPEN')
  683. ]]></programlisting>
  684. </example>
  685. </sect3>
  686. <sect3 id="zend.db.adapter.write.delete">
  687. <title>Supprimer des enregistrements</title>
  688. <para>Il est possible de supprimer des enregistrements dans une table. La méthode <code>delete()</code> est
  689. faite pour cela. Elle accepte deux paramètres, le premier est une chaîne désignant la table.</para>
  690. <para>Le second paramètre est une chaîne contenant l'expression SQL utilisée comme critère pour effacer les
  691. enregistrements. Les valeurs de cette expression de sont pas échappées automatiquement, vous devez donc vous
  692. en occuper le cas échéant. Voyez <xref linkend="zend.db.adapter.quoting" /> pour les méthodes concernant
  693. l'échappement.</para>
  694. <para>La valeur retournée par la méthode <code>delete()</code> est le nombre d'enregistrements affectés
  695. (effacés).</para>
  696. <example id="zend.db.adapter.write.delete.example">
  697. <title>Supprimer des enregistrements</title>
  698. <programlisting role="php"><![CDATA[
  699. $n = $db->delete('bugs', 'bug_id = 3');
  700. ]]></programlisting>
  701. </example>
  702. <para>Si vous ne spécifiez pas le second paramètres, tous les enregistrements de la table seront alors
  703. supprimés.</para>
  704. <para>Si le second paramètre est un tableau de chaînes, alors celles ci seront jointe en une expression SQL,
  705. séparées par l'opération <code>AND</code>.</para>
  706. </sect3>
  707. </sect2>
  708. <sect2 id="zend.db.adapter.quoting">
  709. <title>Échapper des valeurs ou des identifiants</title>
  710. <para>Lorsque vous envoyez des requêtes SQL au SGBD, il est souvent nécessaire d'y inclure des paramètres
  711. dynamiques, PHP. Ceci est risqué car si un des paramètres contient certains caractères, comme l'apostrophe ('),
  712. alors la requête résultante risque d'être mal formée. Par exemple, notez le caractère indésirable dans la
  713. requête suivante : <programlisting role="php"><![CDATA[
  714. $name = "O'Reilly";
  715. $sql = "SELECT * FROM bugs WHERE reported_by = '$name'";
  716. echo $sql;
  717. // SELECT * FROM bugs WHERE reported_by = 'O'Reilly'
  718. ]]></programlisting></para>
  719. <para>Pire encore est le cas où de telles erreurs SQL peuvent être utilisées délibérément par une personne afin
  720. de manipuler la logique de votre requête. Si une personne peut manipuler un paramètre de votre requête, par
  721. exemple via un paramètre HTTP ou une autre méthode, alors il peut y avoir une fuite de données, voire même une
  722. corruption totale de votre base de données. Cette technique très préoccupante de violation de la sécurité d'un
  723. SGBD, est appelée "injection SQL" (voyez <ulink
  724. url="http://en.wikipedia.org/wiki/SQL_Injection">http://en.wikipedia.org/wiki/SQL_Injection</ulink>).</para>
  725. <para>La classe Zend_Db Adapter possède des méthodes adaptées pour vous aider à faire face à de telles
  726. vulnérabilités. La solution proposée est l'échappement de tels caractères (comme la "quote" = ') dans les
  727. valeurs PHP avant leur passage dans la chaîne de requête. Ceci vous protège de l'insertion malveillante ou
  728. involontaires, de caractères spéciaux dans les variables PHP faisant partie d'une requête SQL.</para>
  729. <sect3 id="zend.db.adapter.quoting.quote">
  730. <title>Utilisation de <code>quote()</code></title>
  731. <para>La méthode <code>quote()</code> accepte un seul paramètre, une chaîne de caractère. Elle retourne une
  732. chaîne dont les caractères spéciaux ont été échappés d'une manière convenable en fonction du SGBD
  733. sous-jacent. De plus, la chaîne échappée est entourée d'apostrophes ("<code>'</code>").C'est la valeur
  734. standard de délimitations des chaînes en SQL.</para>
  735. <example id="zend.db.adapter.quoting.quote.example">
  736. <title>Utiliser <code>quote()</code></title>
  737. <programlisting role="php"><![CDATA[
  738. $name = $db->quote("O'Reilly");
  739. echo $name;
  740. // 'O\'Reilly'
  741. $sql = "SELECT * FROM bugs WHERE reported_by = $name";
  742. echo $sql;
  743. // SELECT * FROM bugs WHERE reported_by = 'O\'Reilly'
  744. ]]></programlisting>
  745. </example>
  746. <para>Notez que la valeur de retour contient les apostrophes de délimitation autour de la chaîne. Ceci est
  747. différent de certaines fonctions qui se contentent juste d'échapper les caractères spéciaux, telle que
  748. <ulink url="http://www.php.net/mysqli_real_escape_string">mysql_real_escape_string()</ulink>.</para>
  749. <para>Certaines valeurs en revanche n'ont pas besoin d'être délimitées. Certains SGBDs n'acceptent pas que
  750. les valeurs correspondant à des champs de type entier, soient délimitées. Autrement dit, l'exemple suivant
  751. est erroné dans certaines implémentations de SQL. Nous supposons <code>intColumn</code> ayant un type SQL
  752. <code>INTEGER</code> : <programlisting role="php"><![CDATA[
  753. SELECT * FROM atable WHERE intColumn = '123'
  754. ]]></programlisting></para>
  755. <para>Le second paramètre optionnel de <code>quote()</code> permet de spécifier un type SQL.</para>
  756. <example id="zend.db.adapter.quoting.quote.example-2">
  757. <title>Utiliser <code>quote()</code> avec un type SQL</title>
  758. <programlisting role="php"><![CDATA[
  759. $value = '1234';
  760. $sql = 'SELECT * FROM atable WHERE intColumn = '
  761. . $db->quote($value, 'INTEGER');
  762. ]]></programlisting>
  763. </example>
  764. <para>De plus, chaque classe Zend_Db_Adapter possèdent des constantes représentant les différents type SQL
  765. des SGBDs respectifs qu'elles représentent. Ainsi, les constantes <classname>Zend_Db::INT_TYPE</classname>,
  766. <classname>Zend_Db::BIGINT_TYPE</classname>, et <classname>Zend_Db::FLOAT_TYPE</classname> peuvent vous permettre d'écrire un
  767. code portable entre différents SGBDs.</para>
  768. <para>Zend_Db_Table fournit les types SQL à <code>quote()</code> automatiquement en fonction des colonnes
  769. utilisées par la table référencée.</para>
  770. </sect3>
  771. <sect3 id="zend.db.adapter.quoting.quote-into">
  772. <title>Utilisation de <code>quoteInto()</code></title>
  773. <para>Une autre manière est d'échapper une expression SQL contenant une variable PHP. Vous pouvez utiliser
  774. <code>quoteInto()</code> pour cela. Cette méthode accepte trois arguments. Le premier est la chaîne
  775. représentant l'expression SQL dont les paramètres variables sont remplacés par un joker(<code>?</code>), et
  776. le second argument est la variable PHP à utiliser pour le remplacement du joker.</para>
  777. <para>Le joker est le même symbole que celui utilisé par beaucoup de SGBDs pour la substitution de paramètre
  778. dans une requête préparée.<code>quoteInto()</code> ne fait qu'émuler ce comportement : la méthode ne fait
  779. que remplacer le joker par la valeur PHP, en lui appliquant la méthode <code>quote</code>. De vrais
  780. paramètres de requêtes préparées conservent une réelle isolation entre la requête et ses paramètres.</para>
  781. <example id="zend.db.adapter.quoting.quote-into.example">
  782. <title>Utiliser <code>quoteInto()</code></title>
  783. <programlisting role="php"><![CDATA[
  784. $sql = $db->quoteInto("SELECT * FROM bugs WHERE reported_by = ?",
  785. "O'Reilly");
  786. echo $sql;
  787. // SELECT * FROM bugs WHERE reported_by = 'O\'Reilly'
  788. ]]></programlisting>
  789. </example>
  790. <para>Le troisième paramètre optionnel s'utilise comme avec la méthode <code>quote</code>. Il sert à
  791. spécifier un type SQL, les types numériques ne sont pas délimités.</para>
  792. <example id="zend.db.adapter.quoting.quote-into.example-2">
  793. <title>Utiliser <code>quoteInto()</code> avec un type SQL</title>
  794. <programlisting role="php"><![CDATA[
  795. $sql = $db->quoteInto("SELECT * FROM bugs WHERE bug_id = ?",
  796. '1234',
  797. 'INTEGER');
  798. echo $sql;
  799. // SELECT * FROM bugs WHERE reported_by = 1234
  800. ]]></programlisting>
  801. </example>
  802. </sect3>
  803. <sect3 id="zend.db.adapter.quoting.quote-identifier">
  804. <title>Utilisation de <code>quoteIdentifier()</code></title>
  805. <para>Les valeurs ne sont pas les seuls données qui peuvent être dynamiques dans une requête SQL,et donc
  806. passées par des variables PHP. Les noms des tables, des colonnes, ou tout autre identifiant SQL spécial de
  807. la requête peuvent aussi être dynamiques. En général, les identifiant spéciaux d'une requête ont une syntaxe
  808. identique à celle des variables PHP : pas d'espaces dans les noms, certains autres caractères interdits, la
  809. ponctuation est interdite, etc... Aussi, les identifiant ne peuvent valoir certaines valeurs de mots
  810. réservés : une table ne peut s'appeler "FROM". Il se peut donc que vous ayez besoin aussi d'échapper des
  811. paramètres voués à être substitués à des identifiant dans la requête SQL, et non plus à des valeurs.</para>
  812. <para>Le langage SQL possède une caractéristique appelée <emphasis>identifiant délimités</emphasis>. Si vous
  813. entourez un identifiant SQL dans un type spécial de délimiteurs, alors vous pouvez écrire des requêtes qui
  814. auraient été invalides autrement. Ainsi, vous pouvez inclure des espaces, de la ponctuation ou des
  815. caractères internationaux dans vos identifiant, et aussi utiliser des mots réservés.</para>
  816. <para>La méthode <code>quoteIdentifier()</code> fonctionne comme <code>quote()</code>, mais elle utilise un
  817. caractère de délimitation spécial, en fonction du SGBD sous-jacent. Par exemple, le standard SQL spécifie
  818. des doubles quotes (<code>"</code>) et beaucoup de SGBDs utilisent ceci. MySQL utilise les apostrophes
  819. inverses (back-quotes) (<code>`</code>) par défaut. Les caractères spéciaux sont aussi échappés.</para>
  820. <example id="zend.db.adapter.quoting.quote-identifier.example">
  821. <title>Utiliser quoteIdentifier()</title>
  822. <programlisting role="php"><![CDATA[
  823. // nous possédons une table ayant un nom correspondant
  824. // à un mot reservé en SQL
  825. $tableName = $db->quoteIdentifier("order");
  826. $sql = "SELECT * FROM $tableName";
  827. echo $sql
  828. // SELECT * FROM "order"
  829. ]]></programlisting>
  830. </example>
  831. <para>Les identifiant SQL délimités sont sensibles à la casse. Vous devriez toujours utiliser la casse telle
  832. qu'elle est utilisée dans votre base de données (nom des tables, des colonnes ...).</para>
  833. <para>Dans les cas où le SQL est généré à l'intérieur des classes <classname>Zend_Db</classname>, alors les
  834. identifiant SQL seront automatiquement échappés. Vous pouvez changer ce comportement avec l'option
  835. <classname>Zend_Db::AUTO_QUOTE_IDENTIFIERS</classname>.Spécifiez la lors de l'instanciation de l'adaptateur. Voyez
  836. <xref linkend="zend.db.adapter.connecting.parameters.example2" />.</para>
  837. </sect3>
  838. </sect2>
  839. <sect2 id="zend.db.adapter.transactions">
  840. <title>Gérer les transactions dans une base de données</title>
  841. <para>Les bases de données définissent les transactions comme étant des unités logiques de travail qui peuvent
  842. êtres validées ("commit") ou annulées ("rollback") en tant qu'une seule opération, même sur de multiples tables.
  843. Toutes les requêtes aux bases de données sont considérées comme faisant partie d'une transaction, même si le
  844. driver de base de données fait ceci implicitement. Ceci s'appelle le mode <emphasis>auto-commit</emphasis>, dans
  845. lequel le driver de base de données créer une transaction pour chaque requête exécutée et la valide. Par défaut
  846. toutes les classes <classname>Zend_Db_Adapter</classname> fonctionnent en mode auto-commit.</para>
  847. <para>Vous pouvez manuellement spécifier lorsque vous voulez démarrer une transaction. Vous contrôler ainsi
  848. combien de requêtes doivent y être exécutées, et valider ou annuler ce groupe de requêtes. Utilisez
  849. <code>beginTransaction()</code> pour démarrer une transaction. Toutes les requêtes suivantes seront alors
  850. exécutées dans cette transaction avant que vous ne l'annuliez, ou validiez.</para>
  851. <para>Pour terminer une transaction, utilisez les méthodes <code>commit()</code> ou <code>rollBack()</code>.
  852. <code>commit()</code> validera et appliquera les changements de la transaction au SGBD, ils deviendront alors
  853. visibles dans les autres transactions.</para>
  854. <para><code>rollBack()</code> fait le contraire : elle annule les changements qu'ont générés les requêtes dans
  855. la transaction. L'annulation n'a aucun effet sur les changements qui ont été opérés par d'autres transactions
  856. parallèles.</para>
  857. <para>Après qu'une transaction soit terminées, <classname>Zend_Db_Adapter</classname> retourne en mode auto-commit jusqu'à
  858. un nouvel appel à <code>beginTransaction()</code>.</para>
  859. <example id="zend.db.adapter.transactions.example">
  860. <title>Manipuler les transactions pour assurer l'intégrité de la logique</title>
  861. <programlisting role="php"><![CDATA[
  862. // Démarre explicitement une transaction.
  863. $db->beginTransaction();
  864. try {
  865. // Essaye d'executer une ou plusieurs requêtes :
  866. $db->query(...);
  867. $db->query(...);
  868. $db->query(...);
  869. // Si toutes ont réussi, valide les changements en une seule passe.
  870. $db->commit();
  871. } catch (Exception $e) {
  872. // Si une des requête s'est mal déroulée, alors nous voulons
  873. // annuler les changements de toutes les requêtes faisant partie
  874. // de la transaction, même celles qui se sont bien déroulées.
  875. // Tous les changements sont annulés d'un seul coup.
  876. $db->rollBack();
  877. echo $e->getMessage();
  878. }
  879. ]]></programlisting>
  880. </example>
  881. </sect2>
  882. <sect2 id="zend.db.adapter.list-describe">
  883. <title>Lister et décrire les tables</title>
  884. <para>La méthode <code>listTables()</code> retourne un tableau de chaînes décrivant les tables de la base de
  885. données courante.</para>
  886. <para>La méthode <code>describeTable()</code> retourne un tableau associatif de métadonnées sur une table.
  887. Spécifiez en le nom en paramètre. Le second paramètre est optionnel et définit la base de données à utiliser,
  888. comme par exemple si aucune n'a été sélectionnée précédemment.</para>
  889. <para>Les clés de ce tableau représentent les noms des colonnes, les valeurs sont un tableau avec les clés
  890. suivantes :</para>
  891. <table cellpadding="5" frame="all" id="zend.db.adapter.list-describe.metadata">
  892. <title>Champs de métadonnées retournés par <code>describeTable()</code></title>
  893. <tgroup align="left" cols="3" colsep="1" rowsep="1">
  894. <thead>
  895. <row>
  896. <entry>clé</entry>
  897. <entry>type</entry>
  898. <entry>description</entry>
  899. </row>
  900. </thead>
  901. <tbody>
  902. <row>
  903. <entry>SCHEMA_NAME</entry>
  904. <entry>(chaîne)</entry>
  905. <entry>Nom de la base de données dans laquelle la table existe.</entry>
  906. </row>
  907. <row>
  908. <entry>TABLE_NAME</entry>
  909. <entry>(chaîne)</entry>
  910. <entry>Nom de la table dans laquelle la colonne existe.</entry>
  911. </row>
  912. <row>
  913. <entry>COLUMN_NAME</entry>
  914. <entry>(chaîne)</entry>
  915. <entry>Nom de la colonne.</entry>
  916. </row>
  917. <row>
  918. <entry>COLUMN_POSITION</entry>
  919. <entry>(entier)</entry>
  920. <entry>Position de la colonne dans la table.</entry>
  921. </row>
  922. <row>
  923. <entry>DATA_TYPE</entry>
  924. <entry>(chaîne)</entry>
  925. <entry>Nom du type de données tel que renvoyé par le SGBD.</entry>
  926. </row>
  927. <row>
  928. <entry>DEFAULT</entry>
  929. <entry>(chaîne)</entry>
  930. <entry>Valeur par défaut de la colonne, si une existe.</entry>
  931. </row>
  932. <row>
  933. <entry>NULLABLE</entry>
  934. <entry>(booléen)</entry>
  935. <entry><code>true</code> si la colonne accepte la valeur SQL '<code>NULL</code>',
  936. <code>false</code> sinon.</entry>
  937. </row>
  938. <row>
  939. <entry>LENGTH</entry>
  940. <entry>(entier)</entry>
  941. <entry>Longueur ou taille de la colonne telle que reportée par le SGBD.</entry>
  942. </row>
  943. <row>
  944. <entry>SCALE</entry>
  945. <entry>(entier)</entry>
  946. <entry>Échelle du type SQL NUMERIC ou DECIMAL.</entry>
  947. </row>
  948. <row>
  949. <entry>PRECISION</entry>
  950. <entry>(entier)</entry>
  951. <entry>Précision du type SQL NUMERIC ou DECIMAL.</entry>
  952. </row>
  953. <row>
  954. <entry>UNSIGNED</entry>
  955. <entry>(booléen)</entry>
  956. <entry><code>true</code> si le type est un entier non signé, défini par UNSIGNED.</entry>
  957. </row>
  958. <row>
  959. <entry>PRIMARY</entry>
  960. <entry>(booléen)</entry>
  961. <entry><code>true</code> si la colonne fait partie d'une clé primaire.</entry>
  962. </row>
  963. <row>
  964. <entry>PRIMARY_POSITION</entry>
  965. <entry>(entier)</entry>
  966. <entry>Position de la colonne dans la clé primaire.</entry>
  967. </row>
  968. <row>
  969. <entry>IDENTITY</entry>
  970. <entry>(booléen)</entry>
  971. <entry><code>true</code> si la colonne utilise une valeur auto-générée.</entry>
  972. </row>
  973. </tbody>
  974. </tgroup>
  975. </table>
  976. <note>
  977. <title>A quoi correspond le champs de métadonnées "IDENTITY" en fonction du SGBD ?</title>
  978. <para>Le champs de métadonnées "IDENTITY" a été choisi en tant que terme idiomatique pour représenter une
  979. relation de substitution de clés. Ce champ est généralement connu par les valeurs suivantes :</para>
  980. <itemizedlist>
  981. <listitem>
  982. <para><code>IDENTITY</code> - DB2, MSSQL</para>
  983. </listitem>
  984. <listitem>
  985. <para><code>AUTO_INCREMENT</code> - MySQL</para>
  986. </listitem>
  987. <listitem>
  988. <para><code>SERIAL</code> - PostgreSQL</para>
  989. </listitem>
  990. <listitem>
  991. <para><code>SEQUENCE</code> - Oracle</para>
  992. </listitem>
  993. </itemizedlist>
  994. </note>
  995. <para>Si aucune table ne correspond à votre demande, alors <code>describeTable()</code> retourne un tableau
  996. vide.</para>
  997. </sect2>
  998. <sect2 id="zend.db.adapter.closing">
  999. <title>Fermer une connexion</title>
  1000. <para>Normalement, il n'est pas nécessaire de fermer explicitement sa connexion. PHP nettoie automatiquement les
  1001. ressources laissées ouvertes en fin de traitement. Les extensions des SGBDs ferment alors les connexions
  1002. respectives pour les ressources détruites par PHP.</para>
  1003. <para>Cependant, il se peut que vous trouviez utile de fermer la connexion manuellement. Vous pouvez alors
  1004. utiliser la méthode de l'adaptateur <code>closeConnection()</code> afin de fermer explicitement la connexion
  1005. vers le SGBD.</para>
  1006. <para>A partir de la version 1.7.2, vous pouvez vérifier si vous êtes actuellement connecté au serveur SGBD
  1007. grâce à la méthode <code>isConnected()</code>. Ceci correspond à une ressource de connexion qui a été initiée et
  1008. qui n'est pas close. Cette fonction ne permet pas actuellement de tester la fermeture de la connexion au niveau
  1009. du SGBD par exemple. Cette fonction est utilisée en interne pour fermer la connexion. Elle vous permet entre
  1010. autres de fermer plusieurs fois une connexion sans erreurs. C'était déjà le cas avant la version 1.7.2 pour les
  1011. adaptateurs de type PDO mais pas pour les autres.</para>
  1012. <example id="zend.db.adapter.closing.example">
  1013. <title>Fermer une connexion à un SGBD</title>
  1014. <programlisting role="php"><![CDATA[
  1015. $db->closeConnection();
  1016. ]]></programlisting>
  1017. </example>
  1018. <note>
  1019. <title>Zend_Db supporte-t-il les connexions persistantes ?</title>
  1020. <para>L'utilisation de connexions persistantes n'est pas supporté, ni conseillé, par le composant
  1021. <classname>Zend_Db</classname>.</para>
  1022. <para>Utiliser des connexions persistantes peut mener à un trop grand nombre de connexions en attente
  1023. (idle), ce qui causera plus de problème que cela n'est sensé en résoudre.</para>
  1024. <para>Les connexions aux bases de données possède un état. Dans cet état sont mémorisés des objets propres
  1025. au SGBD. Par exemples des verrous, des variables utilisateur, des tables temporaires, des informations sur
  1026. les requêtes récentes, les derniers enregistrements affectés, les dernières valeurs auto-générées, etc. Avec
  1027. des connexions persistantes, il se peut que vous accédiez à des données ne faisant pas partie de votre
  1028. session de travail avec le SGBD, ce qui peut s'avérer dangereux.</para>
  1029. </note>
  1030. </sect2>
  1031. <sect2 id="zend.db.adapter.other-statements">
  1032. <title>Exécuter des requêtes sur le driver directement</title>
  1033. <para>Il peut y avoir des cas où vous souhaitez accéder directement à la connexion 'bas niveau', sous
  1034. <classname>Zend_Db_Adapter</classname>.</para>
  1035. <para>Par exemple, toute requête effectuée par <classname>Zend_Db</classname> est préparée, et exécutée. Cependant,
  1036. certaines caractéristiques des bases de données ne sont pas compatibles avec les requêtes préparées. Par
  1037. exemple, des requêtes du type CREATE ou ALTER ne peuvent pas être préparées sous MySQL. De même, les requêtes
  1038. préparées ne bénéficient pas du <ulink url="http://dev.mysql.com/doc/refman/5.1/en/query-cache-how.html"> cache
  1039. de requêtes</ulink>, avant MySQL 5.1.17.</para>
  1040. <para>La plupart des extensions PHP pour les bases de données proposent une méthode permettant d'envoyer une
  1041. requête directe, sans préparation. Par exemple, PDO propose pour ceci la méthode <code>exec()</code>. Vous
  1042. pouvez récupérer l'objet de connexion "bas niveau" grâce à la méthode de l'adaptateur
  1043. <code>getConnection()</code>.</para>
  1044. <example id="zend.db.adapter.other-statements.example">
  1045. <title>Envoyer une requête directe dans un adaptateur PDO</title>
  1046. <programlisting role="php"><![CDATA[
  1047. $result = $db->getConnection()->exec('DROP TABLE bugs');
  1048. ]]></programlisting>
  1049. </example>
  1050. <para>De la même manière, vous pouvez accéder à toutes les propriétés ou méthodes de l'objet "bas niveau",
  1051. utilisé par <classname>Zend_Db</classname>. Attention toutefois en utilisant ce procédé, vous risquez de rendre votre
  1052. application dépendante du SGBD qu'elle utilise, en manipulant des méthodes propres à l'extension
  1053. utilisée.</para>
  1054. <para>Dans de futures versions de <classname>Zend_Db</classname>, il sera possible d'ajouter des méthodes pour des
  1055. fonctionnalités communes aux extensions de bases de données de PHP. Ceci ne rompra pas la compatibilité.</para>
  1056. </sect2>
  1057. <sect2 id="zend.db.adapter.server-version">
  1058. <title>Récupérer la version du serveur SGBD</title>
  1059. <para>A partir de la version 1.7.2, vous pouvez récupérer la version du serveur avec le style de syntaxe PHP ce
  1060. qui permet d'utiliser <code>version_compare()</code>. Si cette information n'est pas disponible, vous recevrez
  1061. un <code>null</code>.</para>
  1062. <example id="zend.db.adapter.server-version.example">
  1063. <title>Vérifier la version du serveur avant de lancer une requête</title>
  1064. <programlisting role="php"><![CDATA[
  1065. $version = $db->getServerVersion();
  1066. if (!is_null($version)) {
  1067. if (version_compare($version, '5.0.0', '>=')) {
  1068. // faire quelquechose
  1069. } else {
  1070. // faire autre chose
  1071. }
  1072. } else {
  1073. // impossible de lire la version du serveur
  1074. }
  1075. ]]></programlisting>
  1076. </example>
  1077. </sect2>
  1078. <sect2 id="zend.db.adapter.adapter-notes">
  1079. <title>Notes sur des adaptateur spécifiques</title>
  1080. <para>Cette section liste des différences entre les adaptateurs, que vous devriez considérer.</para>
  1081. <sect3 id="zend.db.adapter.adapter-notes.ibm-db2">
  1082. <title>IBM DB2</title>
  1083. <itemizedlist>
  1084. <listitem>
  1085. <para>Passez le paramètre 'Db2' à la méthode <code>factory()</code>.</para>
  1086. </listitem>
  1087. <listitem>
  1088. <para>Cet adaptateur utilise l'extension PHP ibm_db2.</para>
  1089. </listitem>
  1090. <listitem>
  1091. <para>IBM DB2 supporte les séquences et les clés auto-incrémentées. Les arguments de
  1092. <code>lastInsertId()</code> sont donc optionnels. Si vous ne passez pas de paramètres, alors
  1093. l'adaptateur retourne la dernière valeur de clé auto- incrémentée. Sinon, il retourne la dernière
  1094. valeur de la séquence passée en paramètre, en se référant à la convention
  1095. '<emphasis>table</emphasis>_<emphasis>colonne</emphasis>_seq'.</para>
  1096. </listitem>
  1097. </itemizedlist>
  1098. </sect3>
  1099. <sect3 id="zend.db.adapter.adapter-notes.mysqli">
  1100. <title>MySQLi</title>
  1101. <itemizedlist>
  1102. <listitem>
  1103. <para>Passez le paramètre 'Mysqli' à la méthode <code>factory()</code>.</para>
  1104. </listitem>
  1105. <listitem>
  1106. <para>Cet adaptateur utilise l'extension PHP mysqli.</para>
  1107. </listitem>
  1108. <listitem>
  1109. <para>MySQL ne supporte pas les séquences, donc <code>lastInsertId()</code> ignore tout paramètre
  1110. qu'on lui passe. Elle retourne toujours la valeur de la dernière clé auto-incrémentée.
  1111. <code>lastSequenceId()</code>, elle, retourne toujours <code>null</code>.</para>
  1112. </listitem>
  1113. </itemizedlist>
  1114. </sect3>
  1115. <sect3 id="zend.db.adapter.adapter-notes.oracle">
  1116. <title>Oracle</title>
  1117. <itemizedlist>
  1118. <listitem>
  1119. <para>Passez le paramètre 'Oracle' à la méthode <code>factory()</code>.</para>
  1120. </listitem>
  1121. <listitem>
  1122. <para>Cet adaptateur utilise l'extension PHP oci8.</para>
  1123. </listitem>
  1124. <listitem>
  1125. <para>Oracle ne supporte pas les clé auto-incrémentées, donc vous devriez spécifier un paramètre de
  1126. séquence à <code>lastInsertId()</code> ou <code>lastSequenceId()</code>.</para>
  1127. </listitem>
  1128. <listitem>
  1129. <para>L'extension Oracle ne supporte pas les paramètres positionnés (?). Vous devez utiliser des
  1130. paramètres nommés (:name).</para>
  1131. </listitem>
  1132. <listitem>
  1133. <para>Actuellement l'option <classname>Zend_Db::CASE_FOLDING</classname> n'est pas supportée par l'adaptateur
  1134. Oracle. Pour l'utiliser, vous devez utiliser l'adaptateur basé sur PDO et OCI.</para>
  1135. </listitem>
  1136. <listitem>
  1137. <para>Par défaut les champs LOB ("Large Objet Binaire") sont retournés sous la forme d'objets
  1138. OCI-Lob. Vous pouvez les récupérer sous forme de chaînes pour toutes les requêtes en utilisant
  1139. l'option de driver <code>'lob_as_string'</code> ou pour une requête en particulier en utilisant la
  1140. méthode <code>setLobAsString(boolean)</code> de l'adaptateur ou de l'objet statement.</para>
  1141. </listitem>
  1142. </itemizedlist>
  1143. </sect3>
  1144. <sect3 id="zend.db.adapter.adapter-notes.pdo-ibm">
  1145. <title>PDO pour IBM DB2 et Informix Dynamic Server (IDS)</title>
  1146. <itemizedlist>
  1147. <listitem>
  1148. <para>Passez le paramètre 'PDO_Ibm' à la méthode <code>factory()</code>.</para>
  1149. </listitem>
  1150. <listitem>
  1151. <para>Cet adaptateur utilise les extensions PHP pdo et pdo_ibm.</para>
  1152. </listitem>
  1153. <listitem>
  1154. <para>Vous devez possédez l'extension PDO_IBM en version 1.2.2 minimum. Si ce n'est pas le cas, vous
  1155. devrez la mettre à jour via PECL.</para>
  1156. </listitem>
  1157. </itemizedlist>
  1158. </sect3>
  1159. <sect3 id="zend.db.adapter.adapter-notes.pdo-mssql">
  1160. <title>PDO Microsoft SQL Server</title>
  1161. <itemizedlist>
  1162. <listitem>
  1163. <para>Passez le paramètre 'PDO_Mssql' à la méthode <code>factory()</code>.</para>
  1164. </listitem>
  1165. <listitem>
  1166. <para>Cet adaptateur utilise les extensions PHP pdo et pdo_mssql.</para>
  1167. </listitem>
  1168. <listitem>
  1169. <para>Microsoft SQL Server ne supporte pas les séquences, ainsi <code>lastInsertId()</code> ignore
  1170. les paramètres qu'on lui passe et retourne toujours la valeur de la dernière clé auto-incrémentée.
  1171. <code>lastSequenceId()</code> retourne toujours <code>null</code>.</para>
  1172. </listitem>
  1173. <listitem>
  1174. <para>Si vous travaillez avec des chaînes Unicode avec un encodage autre que UCS-2 (comme UTF-8),
  1175. vous devrez peut-être réaliser une conversion dans votre code d'application ou stocker les données
  1176. dans un champs binaire. Reportez vous à la <ulink url="http://support.microsoft.com/kb/232580"> base
  1177. de connaissance Microsoft</ulink> pour plus d'informations.</para>
  1178. </listitem>
  1179. <listitem>
  1180. <para>Zend_Db_Adapter_Pdo_Mssql met <code>QUOTED_IDENTIFIER</code> à ON dès que la connexion a été
  1181. effectuée. Le driver utilise donc le délimiteur d'identifiant SQL <code>"</code> au lieu de son
  1182. délimiteur habituel.</para>
  1183. </listitem>
  1184. <listitem>
  1185. <para>Vous pouvez spécifier la clé <code>pdoType</code> dans le tableau d'options de construction de
  1186. l'adaptateur. La valeur peut être "mssql" (défaut), "dblib", "freetds", ou "sybase". Cette option
  1187. affecte la syntaxe du préfixe DSN que l'adaptateur utilisera. "freetds" et "sybase" impliquent un
  1188. préfixe "sybase:", qui est utilisé par les librairies <ulink
  1189. url="http://www.freetds.org/">FreeTDS</ulink>.Voyez aussi <ulink
  1190. url="http://www.php.net/manual/en/ref.pdo-dblib.connection.php">
  1191. http://www.php.net/manual/en/ref.pdo-dblib.connection.php</ulink> pour plus d'informations sur les
  1192. DSN pour ce driver.</para>
  1193. </listitem>
  1194. </itemizedlist>
  1195. </sect3>
  1196. <sect3 id="zend.db.adapter.adapter-notes.pdo-mysql">
  1197. <title>PDO MySQL</title>
  1198. <itemizedlist>
  1199. <listitem>
  1200. <para>Passez le paramètre 'PDO_Mysql' à la méthode <code>factory()</code>.</para>
  1201. </listitem>
  1202. <listitem>
  1203. <para>Cet adaptateur utilise les extensions PHP pdo et pdo_mysql.</para>
  1204. </listitem>
  1205. <listitem>
  1206. <para>MySQL ne supporte pas les séquences, ainsi <code>lastInsertId()</code> ignore les paramètres
  1207. qu'on lui passe et retourne toujours la valeur de la dernière clé auto-incrémentée.
  1208. <code>lastSequenceId()</code> retourne toujours <code>null</code>.</para>
  1209. </listitem>
  1210. </itemizedlist>
  1211. </sect3>
  1212. <sect3 id="zend.db.adapter.adapter-notes.pdo-oci">
  1213. <title>PDO Oracle</title>
  1214. <itemizedlist>
  1215. <listitem>
  1216. <para>Passez le paramètre 'PDO_Oci' à la méthode <code>factory()</code>.</para>
  1217. </listitem>
  1218. <listitem>
  1219. <para>Cet adaptateur utilise les extensions PHP pdo et pdo_oci.</para>
  1220. </listitem>
  1221. <listitem>
  1222. <para>Oracle ne supporte pas les clé auto-incrémentées, donc vous devriez spécifier un paramètre de
  1223. séquence à <code>lastInsertId()</code> ou <code>lastSequenceId()</code>.</para>
  1224. </listitem>
  1225. </itemizedlist>
  1226. </sect3>
  1227. <sect3 id="zend.db.adapter.adapter-notes.pdo-pgsql">
  1228. <title>PDO PostgreSQL</title>
  1229. <itemizedlist>
  1230. <listitem>
  1231. <para>Passez le paramètre 'PDO_Pgsql' à la méthode <code>factory()</code>.</para>
  1232. </listitem>
  1233. <listitem>
  1234. <para>Cet adaptateur utilise les extensions PHP pdo et pdo_pgsql.</para>
  1235. </listitem>
  1236. <listitem>
  1237. <para>PostgreSQL supporte les séquences et les clés auto-incrémentées. Les arguments de
  1238. <code>lastInsertId()</code> sont donc optionnels. Si vous ne passez pas de paramètres, alors
  1239. l'adaptateur retourne la dernière valeur de clé auto- incrémentée. Sinon, il retourne la dernière
  1240. valeur de la séquence passée en paramètre, en se référant à la convention
  1241. '<emphasis>table</emphasis>_<emphasis>colonne</emphasis>_seq'.</para>
  1242. </listitem>
  1243. </itemizedlist>
  1244. </sect3>
  1245. <sect3 id="zend.db.adapter.adapter-notes.pdo-sqlite">
  1246. <title>PDO SQLite</title>
  1247. <itemizedlist>
  1248. <listitem>
  1249. <para>Passez le paramètre 'PDO_Sqlite' à la méthode <code>factory()</code>.</para>
  1250. </listitem>
  1251. <listitem>
  1252. <para>Cet adaptateur utilise les extensions PHP pdo et pdo_sqlite.</para>
  1253. </listitem>
  1254. <listitem>
  1255. <para>SQLite ne supporte pas les séquences, ainsi <code>lastInsertId()</code> ignore les paramètres
  1256. qu'on lui passe et retourne toujours la valeur de la dernière clé auto-incrémentée.
  1257. <code>lastSequenceId()</code> retourne toujours <code>null</code>.</para>
  1258. </listitem>
  1259. <listitem>
  1260. <para>Pour vous connecter à une base de données SQLite2, spécifiez le paramètre
  1261. <code>'sqlite2'=&gt;true</code> dans le tableau d'options passé à l'adaptateur, lors de la création
  1262. de l'instance de Pdo_Sqlite Adapter.</para>
  1263. </listitem>
  1264. <listitem>
  1265. <para>Pour vous connecter à une base de données SQLite en mémoire, spécifiez le paramètre
  1266. <code>'dsnprefix'=&gt;':memory:'</code> dans le tableau d'options passé à l'adaptateur, lors de la
  1267. création de l'instance de Pdo_Sqlite Adapter.</para>
  1268. </listitem>
  1269. <listitem>
  1270. <para>Les anciennes versions du driver SQLite pour PHP ne semblent pas supporter les commandes
  1271. PRAGMA nécessaires pour s'assurer que les colonnes ayant un nom court soient utilisées dans les
  1272. résultats. Si vous avez des problèmes, tels que vos enregistrements sont retournés avec une forme
  1273. "nomtable.nomcolonne" lors de vos jointures, vous devriez alors mettre à jour votre version de
  1274. PHP.</para>
  1275. </listitem>
  1276. </itemizedlist>
  1277. </sect3>
  1278. <sect3 id="zend.db.adapter.adapter-notes.firebird">
  1279. <title>Firebird/Interbase</title>
  1280. <itemizedlist>
  1281. <listitem>
  1282. <para>Cet adaptateur utilise l'extension PHP php_interbase.</para>
  1283. </listitem>
  1284. <listitem>
  1285. <para>Firebird/interbase ne supporte pas les clé auto-incrémentées, donc vous devez spécifier un
  1286. paramètre de séquence à <code>lastInsertId()</code> ou <code>lastSequenceId()</code>.</para>
  1287. </listitem>
  1288. <listitem>
  1289. <para>Pour l'instant l'option <classname>Zend_Db::CASE_FOLDING</classname> n'est pas supportée par
  1290. l'adaptateur Firebird/interbase. Tout identificateur non échappé sera automatiquement retourné en
  1291. majuscules.</para>
  1292. </listitem>
  1293. </itemizedlist>
  1294. </sect3>
  1295. </sect2>
  1296. </sect1>