Zend_Db_Adapter.xml 107 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- EN-Revision: 20115 -->
  3. <!-- Reviewed: no -->
  4. <sect1 id="zend.db.adapter">
  5. <title>Zend_Db_Adapter</title>
  6. <para>
  7. <classname>Zend_Db</classname> und die zugehörigen Klassen bieten eine einfache
  8. <acronym>SQL</acronym> Schnittstelle für Zend Framework.
  9. <classname>Zend_Db_Adapter</classname> ist die Basisklasse zur Anbindung einer
  10. <acronym>PHP</acronym> Anwendung an ein <acronym>RDBMS</acronym>. Es gibt für jede
  11. <acronym>RDBMS</acronym> Marke einen eigenen Adapter.
  12. </para>
  13. <para>
  14. Die <classname>Zend_Db</classname> Adapter bilden eine Schnittstelle zu den Hersteller
  15. spezifischen <acronym>PHP</acronym> Erweiterungen und unterstützen dadurch die Entwicklung
  16. einer <acronym>PHP</acronym> Anwendung für verschiedene <acronym>RDBMS</acronym> mit
  17. geringem Aufwand.
  18. </para>
  19. <para>
  20. Die Schnittstellen der Adapterklasse ähneln denen der
  21. <ulink url="http://www.php.net/pdo">PHP Data Objects</ulink> Erweiterung.
  22. <classname>Zend_Db</classname> bietet Adapterklassen für <acronym>PDO</acronym> Treiber der
  23. folgenden <acronym>RDBMS</acronym> Marken:
  24. </para>
  25. <itemizedlist>
  26. <listitem>
  27. <para>
  28. IBM DB2 und Informix Dynamic Server (IDS), verwenden die
  29. <ulink url="http://www.php.net/pdo-ibm">pdo_ibm</ulink> <acronym>PHP</acronym>
  30. Erweiterung
  31. </para>
  32. </listitem>
  33. <listitem>
  34. <para>
  35. MySQL, verwendet die <ulink url="http://www.php.net/pdo-mysql">pdo_mysql</ulink>
  36. <acronym>PHP</acronym> Erweiterung
  37. </para>
  38. </listitem>
  39. <listitem>
  40. <para>
  41. Microsoft <acronym>SQL</acronym> Server, verwendet die <ulink
  42. url="http://www.php.net/pdo-dblib">pdo_dblib</ulink> <acronym>PHP</acronym>
  43. Erweiterung
  44. </para>
  45. </listitem>
  46. <listitem>
  47. <para>
  48. Oracle, verwendet die <ulink url="http://www.php.net/pdo-oci">pdo_oci</ulink>
  49. <acronym>PHP</acronym> Erweiterung
  50. </para>
  51. </listitem>
  52. <listitem>
  53. <para>
  54. PostgreSQL, verwendet die <ulink
  55. url="http://www.php.net/pdo-pgsql">pdo_pgsql</ulink> <acronym>PHP</acronym>
  56. Erweiterung
  57. </para>
  58. </listitem>
  59. <listitem>
  60. <para>
  61. SQLite, verwendet die <ulink url="http://www.php.net/pdo-sqlite">pdo_sqlite</ulink>
  62. <acronym>PHP</acronym> Erweiterung
  63. </para>
  64. </listitem>
  65. </itemizedlist>
  66. <para>
  67. Zusätzlich bietet <classname>Zend_Db</classname> Adapterklassen für die folgenden
  68. <acronym>RDBMS</acronym> Marken, welche eigene <acronym>PHP</acronym> Datenbank
  69. Erweiterungen nutzen:
  70. </para>
  71. <itemizedlist>
  72. <listitem>
  73. <para>
  74. MySQL, mit der <ulink url="http://www.php.net/mysqli">mysqli</ulink>
  75. <acronym>PHP</acronym> Erweiterung
  76. </para>
  77. </listitem>
  78. <listitem>
  79. <para>
  80. Oracle, mit der <ulink url="http://www.php.net/oci8">oci8</ulink>
  81. <acronym>PHP</acronym> Erweiterung
  82. </para>
  83. </listitem>
  84. <listitem>
  85. <para>
  86. IBM DB2 und DB2/i5, mit der <ulink url="http://www.php.net/ibm_db2">ibm_db2</ulink>
  87. <acronym>PHP</acronym> Erweiterung
  88. </para>
  89. </listitem>
  90. <listitem>
  91. <para>
  92. Firebird/Interbase, mit der <ulink
  93. url="http://www.php.net/ibase">php_interbase</ulink> <acronym>PHP</acronym>
  94. Erweiterung
  95. </para>
  96. </listitem>
  97. </itemizedlist>
  98. <note>
  99. <para>
  100. Jeder <classname>Zend_Db</classname> Adapter nutzt eine <acronym>PHP</acronym>
  101. Erweiterung. Die entsprechend <acronym>PHP</acronym> Erweiterung muss in der
  102. <acronym>PHP</acronym> Umgebung aktiviert sein um den <classname>Zend_Db</classname>
  103. Adapter zu nutzen. Zum Beispiel muss bei der Nutzung eines <acronym>PDO</acronym>
  104. <classname>Zend_Db</classname> Adapters sowohl die <acronym>PDO</acronym> Erweiterung,
  105. als auch der <acronym>PDO</acronym> Treiber für die jeweilige <acronym>RDBMS</acronym>
  106. Marke geladen sein.
  107. </para>
  108. </note>
  109. <sect2 id="zend.db.adapter.connecting">
  110. <title>Anbindung einer Datenbank mit einem Adapter</title>
  111. <para>
  112. Dieser Abschnitt beschreibt wie eine Instanz eines Datenbankadapters erzeugt wird. Dies
  113. entspricht der Erzeugung einer Verbindung an ein <acronym>RDBMS</acronym> Server in
  114. einer <acronym>PHP</acronym> Anwendung.
  115. </para>
  116. <sect3 id="zend.db.adapter.connecting.constructor">
  117. <title>Nutzung des Zend_Db Adapter Konstruktors</title>
  118. <para>
  119. Man kann eine Instanz eines Adapters erzeugen, indem man den Konstruktor verwendet.
  120. Ein Adapter Konstruktur benötigt ein Argument, wobei es sich um ein Array mit
  121. Parametern für die Verbindung handelt.
  122. </para>
  123. <example id="zend.db.adapter.connecting.constructor.example">
  124. <title>Nutzung eines Adapter Konstruktors</title>
  125. <programlisting language="php"><![CDATA[
  126. $db = new Zend_Db_Adapter_Pdo_Mysql(array(
  127. 'host' => '127.0.0.1',
  128. 'username' => 'webuser',
  129. 'password' => 'xxxxxxxx',
  130. 'dbname' => 'test'
  131. ));
  132. ]]></programlisting>
  133. </example>
  134. </sect3>
  135. <sect3 id="zend.db.adapter.connecting.factory">
  136. <title>Nutzung der Zend_Db Factory</title>
  137. <para>
  138. Als Alternative zur direkten Nutzung des Konstruktors kann man auch eine Instanz des
  139. Adapters erzeugen indem man die statische Methode
  140. <methodname>Zend_Db::factory()</methodname> nutzt. Diese Methode lädt die
  141. Adapterklasse dynamisch bei Aufruf unter Nutzung von <link
  142. linkend="zend.loader.load.class">Zend_Loader::loadClass()</link>.
  143. </para>
  144. <para>
  145. Das erste Argument ist ein String der den Namen der Adapterklasse enthält. Zum
  146. Beispiel entspricht der String '<classname>Pdo_Mysql</classname>' der Klasse
  147. <classname>Zend_Db_Adapter_Pdo_Mysql</classname>. Das zweite Argument ist das
  148. gleiche Array von Parametern wie bei der Verwendung des Adapter Konstruktors.
  149. </para>
  150. <example id="zend.db.adapter.connecting.factory.example">
  151. <title>Nutzung der Adapter factory() Methode</title>
  152. <programlisting language="php"><![CDATA[
  153. // Wir benötigen das folgende Statement nicht da die
  154. // Zend_Db_Adapter_Pdo_Mysql Datei für uns durch die Factory
  155. // Methode von Zend_Db geladen wird
  156. // require_once 'Zend/Db/Adapter/Pdo/Mysql.php';
  157. // Lädt automatisch die Klasse Zend_Db_Adapter_Pdo_Mysql
  158. // und erzeugt eine Instanz von Ihr.
  159. $db = Zend_Db::factory('Pdo_Mysql', array(
  160. 'host' => '127.0.0.1',
  161. 'username' => 'webuser',
  162. 'password' => 'xxxxxxxx',
  163. 'dbname' => 'test'
  164. ));
  165. ]]></programlisting>
  166. </example>
  167. <para>
  168. Wenn eine eigene Klasse geschrieben wird, die
  169. <classname>Zend_Db_Adapter_Abstract</classname> erweitert aber nicht mit
  170. dem Präfix "<classname>Zend_Db_Adapter</classname>" beginnt, kann die
  171. <methodname>factory()</methodname> Methode verwendet werden um den Adapter zu
  172. Laden wenn der führende Teil der Adapter Klasse mit dem 'adapterNamespace'
  173. Schlüssel im Parameter Array spezifiziert wird.
  174. </para>
  175. <example id="zend.db.adapter.connecting.factory.example2">
  176. <title>Die factory Methode für eine eigene Adapter Klasse verwenden</title>
  177. <programlisting language="php"><![CDATA[
  178. // Wir müssen die Datei der Adapter Klasse nicht laden
  179. // weil Sie für uns durch die Factory Methode von Zend_Db geladen wird
  180. // Die MyProject_Db_Adapter_Pdo_Mysql Klasse automatisch laden
  181. // und eine Instanz von Ihr erstellen.
  182. $db = Zend_Db::factory('Pdo_Mysql', array(
  183. 'host' => '127.0.0.1',
  184. 'username' => 'webuser',
  185. 'password' => 'xxxxxxxx',
  186. 'dbname' => 'test',
  187. 'adapterNamespace' => 'MyProject_Db_Adapter'
  188. ));
  189. ]]></programlisting>
  190. </example>
  191. </sect3>
  192. <sect3 id="zend.db.adapter.connecting.factory-config">
  193. <title>Zend_Config mit Zend_Db_Factory verwenden</title>
  194. <para>
  195. Optional kann jedes Argument der <methodname>factory()</methodname> Methode als
  196. Objekt des Typs <link linkend="zend.config">Zend_Config</link> spezifiziert werden.
  197. </para>
  198. <para>
  199. Wenn das erste Argument ein Config Objekt ist, wird erwartet das es eine Eigenschaft
  200. enthält die <property>adapter</property> heißt und einen String enthält der nach dem
  201. Adapter Basis Klassen Namen benannt ist. Optional kann das Objekt eine Eigenschaft
  202. genannt <property>params</property> enthalten, mit Subeigenschaften korrespondierend
  203. zu den Parameter Namen des Adapters. Das wird nur verwendet wenn das zweite Argument
  204. für die <methodname>factory()</methodname> Methode nicht angegeben wird.
  205. </para>
  206. <example id="zend.db.adapter.connecting.factory.example1">
  207. <title>
  208. Verwenden der Factory Methode des Adapters mit einem Zend_Config Objekt
  209. </title>
  210. <para>
  211. Im Beispiel anbei wird ein <classname>Zend_Config</classname> Objekt von einem
  212. Array erstellt. Die Daten können auch aus einer externen Datei geladen werden
  213. indem Klassen wie zum Beispiel <link
  214. linkend="zend.config.adapters.ini">Zend_Config_Ini</link> oder <link
  215. linkend="zend.config.adapters.xml">Zend_Config_Xml</link> verwendet werden.
  216. </para>
  217. <programlisting language="php"><![CDATA[
  218. $config = new Zend_Config(
  219. array(
  220. 'database' => array(
  221. 'adapter' => 'Mysqli',
  222. 'params' => array(
  223. 'host' => '127.0.0.1',
  224. 'dbname' => 'test',
  225. 'username' => 'webuser',
  226. 'password' => 'secret',
  227. )
  228. )
  229. )
  230. );
  231. $db = Zend_Db::factory($config->database);
  232. ]]></programlisting>
  233. </example>
  234. <para>
  235. Das zweite Argument der <methodname>factory()</methodname> Methode kann ein
  236. assoziatives Array sein das Einträge enthält die den Parameters des Adapters
  237. entsprechen. Dieses Argument ist optional. Wenn das erste Argument vom Typ
  238. <classname>Zend_Config</classname> ist, wird angenommen das es alle Parameter
  239. enthält, und das zweite Argument wird ignoriert.
  240. </para>
  241. </sect3>
  242. <sect3 id="zend.db.adapter.connecting.parameters">
  243. <title>Adapter Parameter</title>
  244. <para>
  245. Die folgende Liste erklärt die gemeinsamen Parameter die von
  246. <classname>Zend_Db</classname> Adapterklassen erkannt werden.
  247. </para>
  248. <itemizedlist>
  249. <listitem>
  250. <para>
  251. <emphasis>host</emphasis>:
  252. Ein String der den Hostname oder die Ip-Adresse des Datenbankservers
  253. beinhaltet. Wenn die Datenbank auf dem gleichen Host wie die
  254. <acronym>PHP</acronym> Anwendung läuft wird 'localhost' oder '127.0.0.1'
  255. verwendet.
  256. </para>
  257. </listitem>
  258. <listitem>
  259. <para>
  260. <emphasis>username</emphasis>:
  261. Konto Kennung zur Authentisierung einer Verbindung zum
  262. <acronym>RDBMS</acronym> Server.
  263. </para>
  264. </listitem>
  265. <listitem>
  266. <para>
  267. <emphasis>password</emphasis>:
  268. Konto Passwort zur Authentisierung einer Verbindung zum
  269. <acronym>RDBMS</acronym> Server.
  270. </para>
  271. </listitem>
  272. <listitem>
  273. <para>
  274. <emphasis>dbname</emphasis>:
  275. Datenbank Name auf dem <acronym>RDBMS</acronym> Server.
  276. </para>
  277. </listitem>
  278. <listitem>
  279. <para>
  280. <emphasis>port</emphasis>:
  281. Einige <acronym>RDBMS</acronym> Server können Netzwerkverbindungen an vom
  282. Administrator spezifizierten Ports akzeptieren. Der Port-Parameter gibt die
  283. Möglichkeit die Portnummer anzugeben, an welche die <acronym>PHP</acronym>
  284. Anwendung verbindet um der Port-Konfiguration des <acronym>RDBMS</acronym>
  285. Servers zu entsprechen.
  286. </para>
  287. </listitem>
  288. <listitem>
  289. <para>
  290. <emphasis>charset</emphasis>: Spezifiziert das Zeichenset das für diese
  291. Verbindung verwendet werden soll.
  292. </para>
  293. </listitem>
  294. <listitem>
  295. <para>
  296. <emphasis>options</emphasis>:
  297. Dieser Parameter ist ein assoziatives Array von Optionen die in allen
  298. <classname>Zend_Db_Adapter</classname> Klassen enthalten sind.
  299. </para>
  300. </listitem>
  301. <listitem>
  302. <para>
  303. <emphasis>driver_options</emphasis>: Dieser Parameter ist ein assoziatives
  304. Array von zusätzlichen Optionen die spezifisch für die angegebene
  305. Datenbankerweiterung sind. Eine typische Anwendung dieses Parameters ist,
  306. Attribute für einen <acronym>PDO</acronym> Treiber zu setzen.
  307. </para>
  308. </listitem>
  309. <listitem>
  310. <para>
  311. <emphasis>adapterNamespace</emphasis>:
  312. Benennt den führenden Teil des Klassen Namens für den Adapter statt
  313. '<classname>Zend_Db_Adapter</classname>'. Dies kann verwendet werden wenn
  314. man die <methodname>factory()</methodname>Methode verwenden muß um eine
  315. nicht von Zend kommende Datenbank Adapter Klasse zu laden.
  316. </para>
  317. </listitem>
  318. </itemizedlist>
  319. <example id="zend.db.adapter.connecting.parameters.example1">
  320. <title>Übergeben der case-folding Option an die factory</title>
  321. <para>
  322. Diese Option kann über die Konstante
  323. <constant>Zend_Db::CASE_FOLDING</constant> angegeben werden. Sie entspricht
  324. dem <constant>ATTR_CASE</constant> Attribut in <acronym>PDO</acronym> und IBM
  325. DB2 Datenbanktreibern und stellt die Schreibweise von String Schlüsseln in
  326. Abfrageergebnissen ein. Die Option kann den Wert
  327. <constant>Zend_Db::CASE_NATURAL</constant> (der Standard),
  328. <constant>Zend_Db::CASE_UPPER</constant> oder
  329. <constant>Zend_Db::CASE_LOWER</constant> annehmen.
  330. </para>
  331. <programlisting language="php"><![CDATA[
  332. $options = array(
  333. Zend_Db::CASE_FOLDING => Zend_Db::CASE_UPPER
  334. );
  335. $params = array(
  336. 'host' => '127.0.0.1',
  337. 'username' => 'webuser',
  338. 'password' => 'xxxxxxxx',
  339. 'dbname' => 'test',
  340. 'options' => $options
  341. );
  342. $db = Zend_Db::factory('Db2', $params);
  343. ]]></programlisting>
  344. </example>
  345. <example id="zend.db.adapter.connecting.parameters.example2">
  346. <title>Übergeben der auto-quoting Option an die factory</title>
  347. <para>
  348. Diese Option kann über die Konstante
  349. <constant>Zend_Db::AUTO_QUOTE_IDENTIFIERS</constant> angegeben werden. Wenn
  350. der Wert <constant>TRUE</constant> (der Standard) ist, werden Bezeichner wie
  351. Tabellennamen, Spaltennamen und auch Aliase in jeder <acronym>SQL</acronym>
  352. Syntax die vom Adapter Objekt generiert wurde begrenzt. Dies macht es einfach
  353. Bezeichner zu verwenden, die <acronym>SQL</acronym> Schlüsselwörter oder
  354. spezielle Zeichen enthalten. Wenn der Wert <constant>FALSE</constant> ist,
  355. werden Bezeichner nicht automatisch begrenzt. Wenn Bezeichner begrenzt werden
  356. müssen, so kann dies über die <methodname>quoteIdentifier()</methodname> Methode
  357. von Hand getan werden.
  358. </para>
  359. <programlisting language="php"><![CDATA[
  360. $options = array(
  361. Zend_Db::AUTO_QUOTE_IDENTIFIERS => false
  362. );
  363. $params = array(
  364. 'host' => '127.0.0.1',
  365. 'username' => 'webuser',
  366. 'password' => 'xxxxxxxx',
  367. 'dbname' => 'test',
  368. 'options' => $options
  369. );
  370. $db = Zend_Db::factory('Pdo_Mysql', $params);
  371. ]]></programlisting>
  372. </example>
  373. <example id="zend.db.adapter.connecting.parameters.example3">
  374. <title>Übergeben von PDO Treiber Optionen an die factory</title>
  375. <programlisting language="php"><![CDATA[
  376. $pdoParams = array(
  377. PDO::MYSQL_ATTR_USE_BUFFERED_QUERY => true
  378. );
  379. $params = array(
  380. 'host' => '127.0.0.1',
  381. 'username' => 'webuser',
  382. 'password' => 'xxxxxxxx',
  383. 'dbname' => 'test',
  384. 'driver_options' => $pdoParams
  385. );
  386. $db = Zend_Db::factory('Pdo_Mysql', $params);
  387. echo $db->getConnection()
  388. ->getAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY);
  389. ]]></programlisting>
  390. </example>
  391. <example id="zend.db.adapter.connecting.parameters.example4">
  392. <title>Übergabe einer Serialisierungs Option an die Factory</title>
  393. <programlisting language="php"><![CDATA[
  394. $options = array(
  395. Zend_Db::ALLOW_SERIALIZATION => false
  396. );
  397. $params = array(
  398. 'host' => '127.0.0.1',
  399. 'username' => 'webuser',
  400. 'password' => 'xxxxxxxx',
  401. 'dbname' => 'test',
  402. 'options' => $options
  403. );
  404. $db = Zend_Db::factory('Pdo_Mysql', $params);
  405. ]]></programlisting>
  406. </example>
  407. </sect3>
  408. <sect3 id="zend.db.adapter.connecting.getconnection">
  409. <title>Verwalten von Lazy Connections</title>
  410. <para>
  411. Die Erzeugung einer Instanz der Adapterklasse stellt nicht gleichzeitig eine
  412. Verbindung zum <acronym>RDBMS</acronym> her. Der Adapter speichert die
  413. Verbindungsparameter und stellt die tatsächliche Verbindung bei Bedarf her, wenn die
  414. erste Anfrage ausgeführt wird. Dies stellt sicher, dass die Erzeugung eines
  415. Adapterobjekts schnell und sparsam ist. Dadurch kann auch dann eine Instanz eines
  416. Adapters erzeugt werden, wenn nicht zwingend eine Datenbankanfrage für die aktuell
  417. gelieferte Darstellung der Anwendung benötigt wird.
  418. </para>
  419. <para>
  420. Wenn der Adapter zwingend eine Verbindung zum <acronym>RDBMS</acronym> herstellen
  421. soll, kann die <methodname>getConnection()</methodname> Methode verwendet werden.
  422. Diese liefert ein Objekt der Verbindung, welches eine Verbindung entsprechend der
  423. verwendeten <acronym>PHP</acronym> Erweiterung repräsentiert. Wenn zum Beispiel
  424. irgendeine der <acronym>PDO</acronym> Adapterklassen verwendet wird, dann liefert
  425. <methodname>getConnection()</methodname> das <acronym>PDO</acronym> Object, nachdem
  426. es als eine live Verbindung zu der entsprechenden Datenbank initialisiert wurde.
  427. </para>
  428. <para>
  429. Es kann nützlich sein eine Verbindung zu erzwingen um jegliche Exceptions
  430. abzufangen, die als Resultat falscher Konto Berechtigungen oder einem anderen Fehler
  431. bei der Verbindung zum <acronym>RDBMS</acronym> auftreten. Diese Exceptions treten
  432. nicht auf, bis die tatsächliche Verbindung hergestellt ist, daher kann es den
  433. Anwendungs-Code vereinfachen, wenn diese Exceptions an einer Stelle bearbeitet
  434. werden, und nicht erst bei der ersten Anfrage.
  435. </para>
  436. <para>
  437. Zusätzlich kann ein Adapter serialisiert werden um Ihn zu speichern, zum Beispiel in
  438. einer Session Variable. Das kann sehr nütlich sein, nicht nur für den Adapter
  439. selbst, sondern auch für andere Objekte die Ihn verwenden, wie ein
  440. <classname>Zend_Db_Select</classname> Objekt. Standardmäßig, ist es Adaptern erlaubt
  441. serialisiert zu werden. Wenn man das nicht will, sollte man die
  442. <constant>Zend_Db::ALLOW_SERIALIZATION</constant> Option mit
  443. <constant>FALSE</constant> übergeben, wie im
  444. Beispiel anbei gezeigt. Um das Prinzip von Lazy Connections zu erlauben, wird der
  445. Adapter sich selbst nicht wiederverbinden wenn er deserialisiert wird. Man muß
  446. <methodname>getConnection()</methodname> selbst aufrufen. Mann kann den Adapter
  447. dazu bringen sich automatisch wieder zu verbinden indem
  448. <constant>Zend_Db::AUTO_RECONNECT_ON_UNSERIALIZE</constant> als Option mit
  449. <constant>TRUE</constant> zum Adapter übergeben wird.
  450. </para>
  451. <example
  452. id="zend.db.adapter.connecting.getconnection.example">
  453. <title>Umgang mit Verbindungs Exceptions</title>
  454. <programlisting language="php"><![CDATA[
  455. try {
  456. $db = Zend_Db::factory('Pdo_Mysql', $parameters);
  457. $db->getConnection();
  458. } catch (Zend_Db_Adapter_Exception $e) {
  459. // Möglicherweise ein fehlgeschlagener login,
  460. // oder die RDBMS läuft möglicherweise nicht
  461. } catch (Zend_Exception $e) {
  462. // Möglicherweise kann factory() die definierte Adapter Klasse nicht laden
  463. }
  464. ]]></programlisting>
  465. </example>
  466. </sect3>
  467. </sect2>
  468. <sect2 id="zend.db.adapter.example-database">
  469. <title>Beispiel Datenbank</title>
  470. <para>
  471. In der Dokumentation für die <classname>Zend_Db</classname> Klassen verwenden wir einige
  472. einfache Tabellen um die Verwendung der Klassen und Methoden zu erläutern. Diese
  473. Beispieltabellen können Informationen für das Bugtracking in einem Softwareprojekt
  474. speichern. Die Datenbank enthält vier Tabellen:
  475. </para>
  476. <itemizedlist>
  477. <listitem>
  478. <para>
  479. <emphasis>accounts</emphasis> speichert Informationen über jeden Benutzer des
  480. Bugtracking Systems.
  481. </para>
  482. </listitem>
  483. <listitem>
  484. <para>
  485. <emphasis>products</emphasis> speichert Informationen über jedes Produkt für das
  486. ein Bug erfasst werden kann.
  487. </para>
  488. </listitem>
  489. <listitem>
  490. <para>
  491. <emphasis>bugs</emphasis> speichert informationen über Bugs, dazu gehört der
  492. derzeitige Status des Bugs, die Person die den Bug berichtet hat, die Person die
  493. den Bug beheben soll und die Person welche die Fehlerbehebung verifizieren soll.
  494. </para>
  495. </listitem>
  496. <listitem>
  497. <para>
  498. <emphasis>bugs_products</emphasis> speichert Beziehungen zwischen Bugs und
  499. Produkten. Dies enthält eine Viele-zu-Viele Beziehung, da ein Bug für mehrere
  500. Produkte relevant sein kann. Und natürlich kann ein Produkt auch mehrere Bugs
  501. enthalten.
  502. </para>
  503. </listitem>
  504. </itemizedlist>
  505. <para>
  506. Der folgende <acronym>SQL</acronym> Daten Definitions Sprache Pseudocode beschreibt die
  507. Tabellen in dieser Beispieldatenbank. Diese Beispieltabellen werden intensiv bei den
  508. automatisierten Unit-Tests für <classname>Zend_Db</classname> verwendet.
  509. </para>
  510. <programlisting language="sql"><![CDATA[
  511. CREATE TABLE accounts (
  512. account_name VARCHAR(100) NOT NULL PRIMARY KEY
  513. );
  514. CREATE TABLE products (
  515. product_id INTEGER NOT NULL PRIMARY KEY,
  516. product_name VARCHAR(100)
  517. );
  518. CREATE TABLE bugs (
  519. bug_id INTEGER NOT NULL PRIMARY KEY,
  520. bug_description VARCHAR(100),
  521. bug_status VARCHAR(20),
  522. reported_by VARCHAR(100) REFERENCES accounts(account_name),
  523. assigned_to VARCHAR(100) REFERENCES accounts(account_name),
  524. verified_by VARCHAR(100) REFERENCES accounts(account_name)
  525. );
  526. CREATE TABLE bugs_products (
  527. bug_id INTEGER NOT NULL REFERENCES bugs,
  528. product_id INTEGER NOT NULL REFERENCES products,
  529. PRIMARY KEY (bug_id, product_id)
  530. );
  531. ]]></programlisting>
  532. <para>
  533. Weiterhin zu beachten ist, dass die 'bugs' Tabelle mehrere Foreign-Key
  534. References zu der 'accounts' Tabelle enthält. Jeder dieser Foreign-Keys kann
  535. auf eine andere Zeile für einen angegebenen Bug in der 'accounts' Tabelle
  536. verweisen.
  537. </para>
  538. <para>
  539. Das unten stehende Diagramm illustriert das physische Datenmodell der Beispieldatenbank.
  540. </para>
  541. <para>
  542. <inlinegraphic width="387" scale="100" align="center" valign="middle"
  543. fileref="figures/zend.db.adapter.example-database.png" format="PNG" />
  544. </para>
  545. </sect2>
  546. <sect2 id="zend.db.adapter.select">
  547. <title>Lesen von Abfrageergebnissen</title>
  548. <para>
  549. Dieser Abschnitt beschreibt Methoden der Adapterklasse mit denen
  550. <acronym>SELECT</acronym> Abfragen ausgeführt werden können um Abfrageergebnisse
  551. abzurufen.
  552. </para>
  553. <sect3 id="zend.db.adapter.select.fetchall">
  554. <title>Holen des kompletten Ergebnisssatzes</title>
  555. <para>
  556. Man kann eine <acronym>SQL</acronym> <acronym>SELECT</acronym> Anfrage ausführen
  557. und alle Ergebnisse auf einmal mit der <methodname>fetchAll()</methodname> Methode
  558. abrufen.
  559. </para>
  560. <para>
  561. Das erste Argument dieser Methode ist ein String der die <acronym>SELECT</acronym>
  562. Anweisung enthält. Als Alternative kann das erste Argument auch ein Objekt der
  563. <link linkend="zend.db.select">Zend_Db_Select</link> Klasse sein. Der Adapter
  564. konvertiert dieses automatisch in einen String der die <acronym>SELECT</acronym>
  565. Anweisung repräsentiert.
  566. </para>
  567. <para>
  568. Das zweite Argument von <methodname>fetchAll()</methodname> ist ein Array von
  569. Werten die Parameterplatzhalter in der <acronym>SQL</acronym> Anweisung ersetzen.
  570. </para>
  571. <example id="zend.db.adapter.select.fetchall.example">
  572. <title>Nutzung von fetchAll()</title>
  573. <programlisting language="php"><![CDATA[
  574. $sql = 'SELECT * FROM bugs WHERE bug_id = ?';
  575. $result = $db->fetchAll($sql, 2);
  576. ]]></programlisting>
  577. </example>
  578. </sect3>
  579. <sect3 id="zend.db.adapter.select.fetch-mode">
  580. <title>Ändern des Fetch Modus</title>
  581. <para>
  582. Standardmäßig gibt <methodname>fetchAll()</methodname> ein Array von Zeilen, jede
  583. als assoziatives Array, zurück. Die Schlüssel von diesem assoziativem Array
  584. entsprechen den Spalten oder Spaltenaliasen wie sie in der SELECT Anfrage benannt
  585. sind.
  586. </para>
  587. <para>
  588. Man kann einen anderen Stil für das Holen der Ergebnisse mit der
  589. <methodname>setFetchMode()</methodname> Methode angeben. Die unterstützten Modi
  590. werden mit folgenden Konstanten identifiziert:
  591. </para>
  592. <itemizedlist>
  593. <listitem>
  594. <para>
  595. <emphasis>Zend_Db::FETCH_ASSOC</emphasis>:
  596. Gibt Daten in einem assoziativem Array zurück. Die Array Schlüssel sind
  597. Strings der Spaltennamen. Dies ist der Standardmodus für
  598. <classname>Zend_Db_Adapter</classname> Klassen.
  599. </para>
  600. <para>
  601. Zu beachten ist, dass wenn die Select-Liste mehr als eine Spalte mit dem
  602. selben Namen enthält, zum Beispiel wenn diese aus verschiedenen Tabellen
  603. durch einem <acronym>JOIN</acronym> bestehen, kann nur einer der Einträge
  604. im assoziativem Array enthalten sein. Wenn der
  605. <acronym>FETCH_ASSOC</acronym> Modus verwandt wird, sollten Spaltenaliase
  606. in der <acronym>SELECT</acronym> Anfrage angegeben werden um sicherzustellen
  607. dass die Namen eindeutige Arrayschlüssel ergeben.
  608. </para>
  609. <para>
  610. Standardmäßig werden die Strings so zurück gegeben wie sie von dem
  611. Datenbanktreiber geliefert werden. Dies entspricht der typischen
  612. Schreibweise der Spaltennamen auf dem <acronym>RDBMS</acronym> Server. Die
  613. Schreibweise dieser Strings kann mit der
  614. <constant>Zend_Db::CASE_FOLDING</constant> Option angegeben werden. Dies
  615. muss bei der Instanziierung des Adapters angegeben werden. Beschreibung
  616. unter <xref linkend="zend.db.adapter.connecting.parameters.example1" />.
  617. </para>
  618. </listitem>
  619. <listitem>
  620. <para>
  621. <emphasis>Zend_Db::FETCH_NUM</emphasis>: Gibt Daten in einem Array von
  622. Arrays zurück. Die Arrays werden über Integer indiziert, entsprechend der
  623. Position der betreffenden Felder in der Select-Liste der Anfrage.
  624. </para>
  625. </listitem>
  626. <listitem>
  627. <para>
  628. <emphasis>Zend_Db::FETCH_BOTH</emphasis>: Gibt ein Array von Arrays zurück.
  629. Die Arrayschlüssel sind sowohl Strings wie beim FETCH_ASSOC Modus, als auch
  630. Integer wie beim FETCH_NUM modus. Zu beachten ist, dass die Anzahl der
  631. Elemente in dem Array doppelt so groß ist, als wenn FETCH_ASSOC oder
  632. FETCH_NUM verwendet worden wäre.
  633. </para>
  634. </listitem>
  635. <listitem>
  636. <para>
  637. <emphasis>Zend_Db::FETCH_COLUMN</emphasis>: Gibt Daten in einem Array von
  638. Werten zurück. Die Werte in jedem Array sind die Werte wie sie in einer
  639. Spalte des Ergebnisses zurück gegeben wurden. Standardmäßig ist die erste
  640. Spalte mit 0 indiziert.
  641. </para>
  642. </listitem>
  643. <listitem>
  644. <para>
  645. <emphasis>Zend_Db::FETCH_OBJ</emphasis>: Gibt Daten in einem Array von
  646. Objekten zurück. Die Standardklasse ist die in <acronym>PHP</acronym>
  647. eingebaute Klasse stdClass. Spalten des Ergebnisses sind als öffentliche
  648. Eigenschaften des Objekts verfügbar.
  649. </para>
  650. </listitem>
  651. </itemizedlist>
  652. <example id="zend.db.adapter.select.fetch-mode.example">
  653. <title>Nutzung von setFetchMode()</title>
  654. <programlisting language="php"><![CDATA[
  655. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  656. $result = $db->fetchAll('SELECT * FROM bugs WHERE bug_id = ?', 2);
  657. // $result ist ein Array von Objekten
  658. echo $result[0]->bug_description;
  659. ]]></programlisting>
  660. </example>
  661. </sect3>
  662. <sect3 id="zend.db.adapter.select.fetchassoc">
  663. <title>Holen eines Ergbnisssatzes als assoziatives Array</title>
  664. <para>
  665. Die <methodname>fetchAssoc()</methodname> Methode gibt Daten in einem Array von
  666. assoziativen Array zurück, egal welcher Wert für den fetch-Modus gesetzt wurde.
  667. </para>
  668. <example id="zend.db.adapter.select.fetchassoc.example">
  669. <title>Nutzung von fetchAssoc()</title>
  670. <programlisting language="php"><![CDATA[
  671. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  672. $result = $db->fetchAssoc('SELECT * FROM bugs WHERE bug_id = ?', 2);
  673. // $result ist ein Array von assoziativen Arrays im Geist von fetch mode
  674. echo $result[0]['bug_description'];
  675. ]]></programlisting>
  676. </example>
  677. </sect3>
  678. <sect3 id="zend.db.adapter.select.fetchcol">
  679. <title>Holen einer einzelnen Spalte eines Ergebnisssatzes</title>
  680. <para>
  681. Die <methodname>fetchCol()</methodname> Methode gibt Daten in einem Array von
  682. Werten zurück, egal welcher Wert für den fetch-Modus gesetzt wurde. Sie gibt nur
  683. die erste Spalte der Anfrage zurück. Alle weiteren Spalten der Anfrage werden
  684. verworfen. Wenn eine andere Spalte als die Erste benötigt wird sollte
  685. <xref linkend="zend.db.statement.fetching.fetchcolumn" /> beachtet werden.
  686. </para>
  687. <example id="zend.db.adapter.select.fetchcol.example">
  688. <title>Nutzung von fetchCol()</title>
  689. <programlisting language="php"><![CDATA[
  690. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  691. $result = $db->fetchCol(
  692. 'SELECT bug_description, bug_id FROM bugs WHERE bug_id = ?', 2);
  693. // Enthält bug_description; bug_id wird nicht zurückgegeben
  694. echo $result[0];
  695. ]]></programlisting>
  696. </example>
  697. </sect3>
  698. <sect3 id="zend.db.adapter.select.fetchpairs">
  699. <title>Holen von Schlüssel-Wert Paaren eines Ergebnisssatzes</title>
  700. <para>
  701. Die <methodname>fetchPairs()</methodname> Methode gibt Daten in einem Array von
  702. Schlüssel-Wert Paaren zurück, einem assoziativen Array mit einem einzelnen Eintrag
  703. pro Zeile. Der Schlüssel dieses assoziativen Arrays wird von der ersten Spalte des
  704. SELECT Ergebnisses genommen. Der Wert wird aus der zweiten Spalte des SELECT
  705. Ergebnisses genommen. Alle weiteren Spalten des Ergebnisses werden verworfen.
  706. </para>
  707. <para>
  708. Die <acronym>SELECT</acronym> Anfrage sollte so gestaltet sein, dass die erste
  709. Spalte nur eindeutige Werte liefert. Wenn doppelte Werte in der ersten Spalte
  710. vorkommen, werden entsprechende Einträge in dem assoziativen Array überschrieben.
  711. </para>
  712. <example id="zend.db.adapter.select.fetchpairs.example">
  713. <title>Nutzung von fetchPairs()</title>
  714. <programlisting language="php"><![CDATA[
  715. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  716. $result = $db->fetchAssoc('SELECT bug_id, bug_status FROM bugs');
  717. echo $result[2];
  718. ]]></programlisting>
  719. </example>
  720. </sect3>
  721. <sect3 id="zend.db.adapter.select.fetchrow">
  722. <title>Holen einer einzelnen Zeile eines Ergebnisssatzes</title>
  723. <para>
  724. Die <methodname>fetchRow()</methodname> Methode gibt Daten entsprechend dem
  725. fetch-Modus zurück, jedoch nur die erste Zeile des Ergebnisssatzes.
  726. </para>
  727. <example id="zend.db.adapter.select.fetchrow.example">
  728. <title>Nutzung von fetchRow()</title>
  729. <programlisting language="php"><![CDATA[
  730. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  731. $result = $db->fetchRow('SELECT * FROM bugs WHERE bug_id = 2');
  732. // Beachte das $result ein einzelnes Objekt ist, und kein Array von Objekten
  733. echo $result->bug_description;
  734. ]]></programlisting>
  735. </example>
  736. </sect3>
  737. <sect3 id="zend.db.adapter.select.fetchone">
  738. <title>Holen eines einzelnen Scalars aus einem Ergebnisssatz</title>
  739. <para>
  740. Die <methodname>fetchOne()</methodname> Methode ist wie eine Kombination von
  741. <methodname>fetchRow()</methodname> mit <methodname>fetchCol()</methodname>, gibt
  742. also nur die erste Zeile des Ergebnisssatze zurück, und von dieser auch nur den
  743. Wert der ersten Spalte. Daher wird nur ein einziger scalarer Wert zurückgegeben,
  744. kein Array und auch kein Objekt.
  745. </para>
  746. <example id="zend.db.adapter.select.fetchone.example">
  747. <title>Nutzung von fetchOne()</title>
  748. <programlisting language="php"><![CDATA[
  749. $result = $db->fetchOne('SELECT bug_status FROM bugs WHERE bug_id = 2');
  750. // this is a single string value
  751. echo $result;
  752. ]]></programlisting>
  753. </example>
  754. </sect3>
  755. </sect2>
  756. <sect2 id="zend.db.adapter.write">
  757. <title>Schreiben von Änderungen in die Datenbank</title>
  758. <para>
  759. Die Adapterklasse kann verwendet werden um neue Daten in die Datenbank zu schreiben oder
  760. bestehende Daten in der Datenbank zu ändern. Dieser Abschnitt beschreibt Methoden für
  761. diese Operationen.
  762. </para>
  763. <sect3 id="zend.db.adapter.write.insert">
  764. <title>Einfügen von Daten</title>
  765. <para>
  766. Neue Zeilen können in die Datenbank mit der <methodname>insert()</methodname>
  767. Methode eingefügt werden. Das erste Argument ist ein String der die Tabelle
  768. benennt, und das zweite Argument ist ein assoziatives Array das den Spaltennamen
  769. Datenwerte zuordnet.
  770. </para>
  771. <example id="zend.db.adapter.write.insert.example">
  772. <title>Einfügen in eine Tabelle</title>
  773. <programlisting language="php"><![CDATA[
  774. $data = array(
  775. 'created_on' => '2007-03-22',
  776. 'bug_description' => 'Etwas falsch',
  777. 'bug_status' => 'NEW'
  778. );
  779. $db->insert('bugs', $data);
  780. ]]></programlisting>
  781. </example>
  782. <para>
  783. Spalten die nicht in dem Array definiert sind, werden nicht an die Datenbank
  784. übergeben. Daher folgen sie den selben Regeln denen eine <acronym>SQL</acronym>
  785. <acronym>INSERT</acronym> Anweisung folgt: wenn die Spalte eine
  786. <acronym>DEFAULT</acronym> Klausel hat, so bekommt die Spalte der neuen Zeile diesen
  787. Wert. Andernfalls behält sie den Status <constant>NULL</constant>.
  788. </para>
  789. <para>
  790. Standardmäßig werden die Daten in dem Array mit Parametern eingefügt. Dies reduziert
  791. das Risiko einiger Typen von Sicherheitsproblemen. Die Werte in dem Array müssen
  792. daher nicht escaped oder quotiert übergeben werden.
  793. </para>
  794. <para>
  795. Einige Werte in dem Array könnten als <acronym>SQL</acronym> Expressions benötigt
  796. werden, in diesem Fall dürfen sie nicht in Anführungszeichen stehen. Standardmäßig
  797. werden alle übergebenen String-Werte als String-literale behandelt. Um anzugeben das
  798. ein Wert eine <acronym>SQL</acronym> Expression ist, und daher nicht quotiert werden
  799. soll, muss der Wert als ein Objekt des Typs <classname>Zend_Db_Expr</classname>
  800. übergeben werden, und nicht als einfacher String.
  801. </para>
  802. <example id="zend.db.adapter.write.insert.example2">
  803. <title>Einfügen von Expressions in eine Tabelle</title>
  804. <programlisting language="php"><![CDATA[
  805. $data = array(
  806. 'created_on' => new Zend_Db_Expr('CURDATE()'),
  807. 'bug_description' => 'Etwas falsch',
  808. 'bug_status' => 'NEW'
  809. );
  810. $db->insert('bugs', $data);
  811. ]]></programlisting>
  812. </example>
  813. </sect3>
  814. <sect3 id="zend.db.adapter.write.lastinsertid">
  815. <title>Abfragen von generierten Werten</title>
  816. <para>
  817. Einige <acronym>RDBMS</acronym> Marken unterstützen Auto-Incrementierung von
  818. Primärschlüsseln. Eine Tabelle die so definiert ist generiert automatisch einen
  819. Primärschlüsselwert während des <acronym>INSERT</acronym>'s einer neuen Zeile. Der
  820. Rückgabewert der <methodname>insert()</methodname> Methode ist
  821. <emphasis>nicht</emphasis> die letzte eingefügte ID, weil die Tabelle keine
  822. Auto-Increment Spalte haben könnte. Statt dessen ist der Rückgabewert die Anzahl der
  823. betroffenen Zeilen (normalerweise 1).
  824. </para>
  825. <para>
  826. Wenn die Tabelle mit einem Auto-Increment Primärschlüssel definiert ist, kann die
  827. <methodname>lastInsertId()</methodname> Methode nach dem INSERT aufgerufen werden.
  828. Diese Methode gibt den letzten generierten Wertim Rahmen der aktuellen
  829. Datenbankverbindung zurück.
  830. </para>
  831. <example id="zend.db.adapter.write.lastinsertid.example-1">
  832. <title>Nutzung von lastInsertId() für einen Auto-Increment Schlüssel</title>
  833. <programlisting language="php"><![CDATA[
  834. $db->insert('bugs', $data);
  835. // Gib den letzten durch eine auto-inkrement Spalte erzeugten Wert zurück
  836. $id = $db->lastInsertId();
  837. ]]></programlisting>
  838. </example>
  839. <para>
  840. Einige <acronym>RDBMS</acronym> Marken unterstützen ein Sequenz-Objekt, welches
  841. eindeutige Werte generiert, die als Primärschlüsselwerte dienen. Um Sequenzen zu
  842. unterstützen, akzeptiert die <methodname>lastInsertId()</methodname> Method zwei
  843. optionale String Argumente. Diese Argumente benennen die Tabelle und die Spalte, in
  844. der Annahme das die Konvention beachtet wurde, dass eine Sequenz mit der Tabelle und
  845. der Spalte benannt wurde, für die sie Werte generiert plus dem Anhang "_seq". Dies
  846. basiert auf der Konvention die von PostgreSQL verwendet wird, wenn Sequenzen für
  847. SERIAL Spalten benannt werden. Zum Beispiel würde eine Tabelle "bugs" mit der
  848. Primärschlüsselspalte "bug_id" eine Sequenz als "bugs_bug_id_seq" benennen.
  849. </para>
  850. <example id="zend.db.adapter.write.lastinsertid.example-2">
  851. <title>Nutzung von lastInsertId() für eine Sequenz</title>
  852. <programlisting language="php"><![CDATA[
  853. $db->insert('bugs', $data);
  854. // Gib den letzten durch die 'bugs_bug_id_seq' Sequenz erstellten Wert zurück
  855. $id = $db->lastInsertId('bugs', 'bug_id');
  856. // Gib, alternativ, den letzten durch die 'bugs_seq' Sequenz
  857. // erstellten Wert zurück
  858. $id = $db->lastInsertId('bugs');
  859. ]]></programlisting>
  860. </example>
  861. <para>
  862. Wenn der Name des Squenz-Objekts nicht dieser Konvention folgt muss die
  863. <methodname>lastSequenceId()</methodname> Methode an Stelle verwendet werden. Diese
  864. Methode benötigt ein String Argument, welches die Sequenz wörtlich benennt.
  865. </para>
  866. <example id="zend.db.adapter.write.lastinsertid.example-3">
  867. <title>Nutzung von lastSequenceId()</title>
  868. <programlisting language="php"><![CDATA[
  869. $db->insert('bugs', $data);
  870. // Gib den letzten durch die 'bugs_id_gen' Sequenz erstellten Wert zurück.
  871. $id = $db->lastSequenceId('bugs_id_gen');
  872. ]]></programlisting>
  873. </example>
  874. <para>
  875. Bei <acronym>RDBMS</acronym> Marken die keine Sequenzen unterstützen, dazu gehören
  876. MySQL, Microsoft <acronym>SQL</acronym> Server und SQLite, werden die Argumente an
  877. die <methodname>lastInsertId()</methodname> Methode ignoriert, und der zurück
  878. gegebene Wert ist der zuletzt für eirgendeine Tabelle während einer
  879. <acronym>INSERT</acronym> Operation generierte Wert innerhalb der aktuellen
  880. Verbindung. Für diese <acronym>RDBMS</acronym> Marken gibt die
  881. <methodname>lastSequenceId()</methodname> Methode immer <constant>NULL</constant>
  882. zurück.
  883. </para>
  884. <note>
  885. <title>Weshalb sollte man nicht "SELECT MAX(id) FROM table" verwenden?</title>
  886. <para>
  887. Manchmal gibt diese Anfrage den zuletzt eingefügten Primärschlüsselwert zurück.
  888. Trotzdem ist diese Technik in einer Umgebung in der mehrere Clients Daten in die
  889. Datenbank einfügen nicht sicher. Es ist möglich, und daher vorherbestimmt
  890. eventuell aufzutreten, das ein anderer Client in dem Augenblick zwischen dem
  891. INSERT deiner Client Anwendung und deiner Anfrage für den
  892. <methodname>MAX(id)</methodname> Wert, eine andere Zeile einfügt. Somit
  893. identifiziert der zurück gegebene Wert nicht die von dir eingefügte Zeile,
  894. sondern die eines anderen Clients. Man kann nie wissen wann dies passiert.
  895. </para>
  896. <para>
  897. Das Nutzen eines starken Transaktions Isolationsmodus wie "repeatable read" kann
  898. das Risiko mindern, aber einige <acronym>RDBMS</acronym> Marken unterstützen
  899. nicht die Transaktions Isolation die hierfür benötigt wird, oder deine
  900. Applikation könnte einen schwächeren Transaktions Isolationsmodus nutzen.
  901. </para>
  902. <para>
  903. Darüberhinaus ist das Nutzen eins Ausdrucks wie "MAX(id)+1" um einen neuen Wert
  904. für den Primärschlüssel zu generiern nict sicher, weil zwei Clients diese
  905. Anfrage gleichzeitig ausführen könnten und damit beide den gleichen Wert für
  906. ihre nächste <acronym>INSERT</acronym> Operation bekommen würden.
  907. </para>
  908. <para>
  909. Alle <acronym>RDBMS</acronym> Marken bieten einen Mechanismus um eindeutige
  910. Werte zu generieren, und um den zuletzt generierten Wert zurück zu geben. Diese
  911. Machanismen funktionieren notwendigerweise außerhalb des Gültigkeitsbereichs
  912. einer Transaktions Isolation, es besteht daher nicht die Möglichkeit das zwei
  913. Clients den selben Wert generieren und es besteht nicht die Möglichkeit das der
  914. Wert, der von einem anderen Client generiert wurde, an die Verbindung deines
  915. Clients, als letzter generierter Wert, gesendet wird.
  916. </para>
  917. </note>
  918. </sect3>
  919. <sect3 id="zend.db.adapter.write.update">
  920. <title>Aktualisieren von Daten</title>
  921. <para>
  922. Zeilen in der Datenbank können mit der <methodname>update()</methodname> Methode
  923. eines Adapters aktualisiert werden. Diese Methode benötigt drei Argumente: Das
  924. Erste ist der Name der Tabelle und das Zweite ist ein assoziatives Array das den zu
  925. Ändernden Spalten neue Werte zuordnet.
  926. </para>
  927. <para>
  928. Die Werte des Datenarrays werden als String Literale behandelt. Beachte
  929. <xref linkend="zend.db.adapter.write.insert" /> für Informationen zur Nutzung von
  930. <acronym>SQL</acronym> Expressions in dem Datenarray.
  931. </para>
  932. <para>
  933. Das dritte Argument ist ein String der aus einer <acronym>SQL</acronym> Expression
  934. besteht, die genutzt wird um Kriterien für die Auswahl der zu ändernden Zeilen zu
  935. bestimmen. Die Werte und Bezeichner in diesem Argument werden nicht escaped oder
  936. quotiert. An dieser Stelle muss darauf geachtet werden das sichergestellt ist, das
  937. dynamischer Inhalt sicher in diesen String eingefügt wird. Unter
  938. <xref linkend="zend.db.adapter.quoting" /> sind Methoden beschrieben die dabei
  939. helfen können.
  940. </para>
  941. <para>
  942. Der Rückgabewert ist die Anzahl der Betroffenen Zeilen der UPDATE Operation.
  943. </para>
  944. <example id="zend.db.adapter.write.update.example">
  945. <title>Aktualisieren von Zeilen</title>
  946. <programlisting language="php"><![CDATA[
  947. $data = array(
  948. 'updated_on' => '2007-03-23',
  949. 'bug_status' => 'FIXED'
  950. );
  951. $n = $db->update('bugs', $data, 'bug_id = 2');
  952. ]]></programlisting>
  953. </example>
  954. <para>
  955. Wenn das dritte Argument ausgelassen wird, werden alle Zeilen der Tabelle mit den
  956. Werten des Datenarrays aktualisiert.
  957. </para>
  958. <para>
  959. Wenn ein Array mit Strings als drittes Argument übergeben wird, werden diese
  960. Strings als eine Expression von Ausdrücken, getrennt von <constant>AND</constant>
  961. Operatoren, zusammengefügt.
  962. </para>
  963. <para>
  964. Wenn man ein Array von Arrays als drittes Argument anbietet, werden die Werte
  965. automatisch in die Schlüssel eingefügt. Diese werden dann zusammen zu Ausdrücken
  966. verbunden, getrennt von <constant>AND</constant> Operatoren.
  967. </para>
  968. <example id="zend.db.adapter.write.update.example-array">
  969. <title>Aktualisieren von Zeilen unter Nutzung eines Arrays von Expressions</title>
  970. <programlisting language="php"><![CDATA[
  971. $data = array(
  972. 'updated_on' => '2007-03-23',
  973. 'bug_status' => 'FIXED'
  974. );
  975. $where[] = "reported_by = 'goofy'";
  976. $where[] = "bug_status = 'OPEN'";
  977. $n = $db->update('bugs', $data, $where);
  978. // Der erstellte SQL Syntax ist:
  979. // UPDATE "bugs" SET "update_on" = '2007-03-23', "bug_status" = 'FIXED'
  980. // WHERE ("reported_by" = 'goofy') AND ("bug_status" = 'OPEN')
  981. ]]></programlisting>
  982. </example>
  983. <example id="zend.db.adapter.write.update.example-arrayofarrays">
  984. <title>Zeilen aktualisieren durch Verwendung von einem Array von Arrays</title>
  985. <programlisting language="php"><![CDATA[
  986. $data = array(
  987. 'updated_on' => '2007-03-23',
  988. 'bug_status' => 'FIXED'
  989. );
  990. $where['reported_by = ?'] = 'goofy';
  991. $where['bug_status = ?'] = 'OPEN';
  992. $n = $db->update('bugs', $data, $where);
  993. // Das resultierende SQL ist:
  994. // UPDATE "bugs" SET "update_on" = '2007-03-23', "bug_status" = 'FIXED'
  995. // WHERE ("reported_by" = 'goofy') AND ("bug_status" = 'OPEN')
  996. ]]></programlisting>
  997. </example>
  998. </sect3>
  999. <sect3 id="zend.db.adapter.write.delete">
  1000. <title>Löschen von Daten</title>
  1001. <para>
  1002. Daten können aus einer Datenbanktabelle mit der <methodname>delete()</methodname>
  1003. Methode gelöscht werden. Diese Methode benötigt zwei Argumente: Das erste ist ein
  1004. String der die Tabelle benennt.
  1005. </para>
  1006. <para>
  1007. Das zweite Argument ist ein String der aus einer <acronym>SQL</acronym> Expression
  1008. besteht, welche Kriterien für die zu löschenden Zeilen enthält. Die Werte und
  1009. Bezeichner in diesem Argument werden nicht escaped quotiert. An dieser Stelle muss
  1010. darauf geachtet werden das sichergestellt ist, das dynamischer Inhalt sicher in
  1011. diesen String eingefügt wird. Unter <xref linkend="zend.db.adapter.quoting" /> sind
  1012. Methoden beschrieben die dabei helfen können.
  1013. </para>
  1014. <para>
  1015. Der Rückgabewert ist die Anzahl der Betroffenen Zeilen der DELETE Operation.
  1016. </para>
  1017. <example id="zend.db.adapter.write.delete.example">
  1018. <title>Löschen von Zeilen</title>
  1019. <programlisting language="php"><![CDATA[
  1020. $n = $db->delete('bugs', 'bug_id = 3');
  1021. ]]></programlisting>
  1022. </example>
  1023. <para>
  1024. Wenn das zweite Argument ausgelassen wird, werden alle Zeilen der Tabelle gelöscht.
  1025. </para>
  1026. <para>
  1027. Wenn ein Array mit Strings als zweites Argument übergeben wird, werden diese
  1028. Strings als eine Expression von Ausdrücken, getrennt von <constant>AND</constant>
  1029. Operatoren, zusammengefügt.
  1030. </para>
  1031. <para>
  1032. Wenn man ein Array von Arrays als zweites Argument übergibt, werden die Werte
  1033. automatisch in die Schlüssel eingefügt. Diese werden dann zusammen zu Ausdrücken
  1034. verbunden, getrennt durch <constant>AND</constant> Operatoren.
  1035. </para>
  1036. </sect3>
  1037. </sect2>
  1038. <sect2 id="zend.db.adapter.quoting">
  1039. <title>Quotierung von Werten und Bezeichnern</title>
  1040. <para>
  1041. Beim Erzeugen von <acronym>SQL</acronym> Anfragen ist es häufig nötig
  1042. <acronym>PHP</acronym> Variablen in die <acronym>SQL</acronym> Expression einzufügen.
  1043. Dies ist riskant, weil der Wert eines <acronym>PHP</acronym> Strings bestimmte Zeichen
  1044. enthalten kann, wie das Anführungszeichen, was zu ungültiger <acronym>SQL</acronym>
  1045. Syntax führen kann. Zum Beispiel, zu beachten ist die ungerade Anzahl der
  1046. Anführungszeichen in der folgenden Anfrage:
  1047. <programlisting language="php"><![CDATA[
  1048. $name = "O'Reilly";
  1049. $sql = "SELECT * FROM bugs WHERE reported_by = '$name'";
  1050. echo $sql;
  1051. // SELECT * FROM bugs WHERE reported_by = 'O'Reilly'
  1052. ]]></programlisting>
  1053. </para>
  1054. <para>
  1055. Noch schlimmer ist das Risiko, dass solche Code-Fehler von einer Person absichtlich
  1056. ausgenutzt werden um die Funktion der Webanwendung zu manipulieren. Wenn der Wert einer
  1057. <acronym>PHP</acronym> Variablen über die Nutzung von <acronym>HTTP</acronym> Parametern
  1058. oder eines anderen Mechanismus gesetzt werden kann, könnte eine Person die
  1059. <acronym>SQL</acronym> Anfragen nutzen um Dinge zu tun, wozu sie nicht gedacht sind, wie
  1060. Daten ausgeben, wozu die Person keine Zugangsberechtigung hat. Dies ist eine ernst zu
  1061. nehmende und weit verbreitete Technik um die Sicherheit einer Anwendung zu verletzen,
  1062. bekannt unter dem Namen "SQL Injection" (siehe <ulink
  1063. url="http://en.wikipedia.org/wiki/SQL_Injection">http://en.wikipedia.org/wiki/SQL_Injection</ulink>).
  1064. </para>
  1065. <para>
  1066. Die <classname>Zend_Db</classname> Adapterklassen bieten bequeme Methoden, die helfen
  1067. die Verletzbarkeit durch <acronym>SQL</acronym> Injection Angriffe im
  1068. <acronym>PHP</acronym> Code zu reduzieren. Die Lösung ist bestimmte Zeichen, wie
  1069. Anführungszeichen, in <acronym>PHP</acronym> Werten zu ersetzen bevor sie in
  1070. <acronym>SQL</acronym> Strings eingefügt werden. Dies schützt sowohl vor versehentlicher
  1071. als auch vor absichtlicher Manipulation von <acronym>SQL</acronym> Strings durch
  1072. <acronym>PHP</acronym> Variablen, die spezielle Zeichen enthalten.
  1073. </para>
  1074. <sect3 id="zend.db.adapter.quoting.quote">
  1075. <title>Nutzung von quote()</title>
  1076. <para>
  1077. Die <methodname>quote()</methodname> Methode benötigt ein Argument, einen skalaren
  1078. String Wert. Sie gibt den Wert mit ersetzten speziellen Zeichen, passend zu dem
  1079. eingesetzten <acronym>RDBMS</acronym>, und umgeben von Stringwertbegrenzern zurück.
  1080. Der Standard <acronym>SQL</acronym> Stringwertbegrenzer ist das einfache
  1081. Anführungszeichen (').
  1082. </para>
  1083. <example id="zend.db.adapter.quoting.quote.example">
  1084. <title>Nutzung von quote()</title>
  1085. <programlisting language="php"><![CDATA[
  1086. $name = $db->quote("O'Reilly");
  1087. echo $name;
  1088. // 'O\'Reilly'
  1089. $sql = "SELECT * FROM bugs WHERE reported_by = $name";
  1090. echo $sql;
  1091. // SELECT * FROM bugs WHERE reported_by = 'O\'Reilly'
  1092. ]]></programlisting>
  1093. </example>
  1094. <para>
  1095. Zu beachten ist, dass der Rückgabewert von <methodname>quote()</methodname> die
  1096. Stringwertbegrenzer enthält. Dies ist ein Unterschied zu anderen Methoden die
  1097. spezielle Zeichen ersetzen, aber keine Stringwertbegrenzer hinzufügen, wie z.B.
  1098. <ulink url="http://www.php.net/mysqli_real_escape_string">mysql_real_escape_string()</ulink>.
  1099. </para>
  1100. <para>
  1101. Es kann notwendig sein Werte in Anführungszeichen zu setzen oder nicht je nach dem
  1102. Kontext des <acronym>SQL</acronym> Datentyps in dem diese verwendet werden. Zum
  1103. Beispiel darf, in einigen RDBMS Typen, ein Integer Wert nicht wie in String in
  1104. Anführungszeichen gesetzt werden, wenn dieser mit einer Integer-Typ Spalte oder
  1105. einem Ausdruck verglichen wird. Anders gesagt ist das folgende in einigen
  1106. <acronym>SQL</acronym> Implementationen ein Fehler, wenn angenommen wird dass
  1107. <property>intColumn</property> einen <acronym>SQL</acronym> Datentyp von
  1108. <constant>INTEGER</constant> besitzt
  1109. <programlisting language="php"><![CDATA[
  1110. SELECT * FROM atable WHERE intColumn = '123'
  1111. ]]></programlisting>
  1112. </para>
  1113. <para>
  1114. Es kann das optionale zweite Argument der <methodname>quote()</methodname> Methode
  1115. verwendet werden um die Verwendung von Anführungszeichen selektiv für den
  1116. spezifizierten <acronym>SQL</acronym> Datentyp auszuwählen.
  1117. </para>
  1118. <example id="zend.db.adapter.quoting.quote.example-2">
  1119. <title>Verwenden von quote() mit einem SQL Typ</title>
  1120. <programlisting language="php"><![CDATA[
  1121. $value = '1234';
  1122. $sql = 'SELECT * FROM atable WHERE intColumn = '
  1123. . $db->quote($value, 'INTEGER');
  1124. ]]></programlisting>
  1125. </example>
  1126. <para>
  1127. Jede <classname>Zend_Db_Adapter</classname> Klasse hat den Namen des nummerischen
  1128. <acronym>SQL</acronym> Datentyps für die respektive Marke von
  1129. <acronym>RDBMS</acronym> codiert. Man kann genauso die Konstanten
  1130. <constant>Zend_Db::INT_TYPE</constant>,
  1131. <constant>Zend_Db::BIGINT_TYPE</constant>, und
  1132. <constant>Zend_Db::FLOAT_TYPE</constant> verwenden um Code in einem mehr
  1133. <acronym>RDBMS</acronym>-unabhängigen Weg zu schreiben.
  1134. </para>
  1135. <para>
  1136. <classname>Zend_Db_Table</classname> definiert <acronym>SQL</acronym> Typen zu
  1137. <methodname>quote()</methodname> automatisch wenn <acronym>SQL</acronym> Abfragen
  1138. erstellt werden die einer Tabellen Schlüssel Spalte entsprechen.
  1139. </para>
  1140. </sect3>
  1141. <sect3 id="zend.db.adapter.quoting.quote-into">
  1142. <title>Nutzung von quoteInto()</title>
  1143. <para>
  1144. Die typischste Anwendung von Quotierung ist das Einfügen von <acronym>PHP</acronym>
  1145. Variablen in eine <acronym>SQL</acronym> Expression oder Anweisung. Die
  1146. <methodname>quoteInto()</methodname> Methode kann verwendet werden um dies in einem
  1147. Schritt zu erledigen. Die Methode benötigt zwei Argumente: Das erste Argument ist
  1148. ein String der ein Platzhaltersymbol (?) enthält, und das zweite
  1149. Argument ist ein Wert oder eine <acronym>PHP</acronym> Variable die den Platzhalter
  1150. ersetzen soll.
  1151. </para>
  1152. <para>
  1153. Das Platzhaltersymbol ist das gleiche Symbol wie es von vielen
  1154. <acronym>RDBMS</acronym> Marken für Lage betreffende Parameter verwendet wird, aber
  1155. die <methodname>quoteInto()</methodname> Methode bildet nur Abfrageparameter nach.
  1156. Die Methode fügt den Wert in den String ein, ersetzt dabei spezielle
  1157. Zeichen und fügt Stringwertbegrenzer ein. Echte Abfrageparameter sorgen für eine
  1158. Trennung von <acronym>SQL</acronym> String und Parametern wenn die Anweisung vom
  1159. <acronym>RDBMS</acronym> Server verarbeitet wird.
  1160. </para>
  1161. <example id="zend.db.adapter.quoting.quote-into.example">
  1162. <title>Nutzung von quoteInto()</title>
  1163. <programlisting language="php"><![CDATA[
  1164. $sql = $db->quoteInto("SELECT * FROM bugs WHERE reported_by = ?", "O'Reilly");
  1165. echo $sql;
  1166. // SELECT * FROM bugs WHERE reported_by = 'O\'Reilly'
  1167. ]]></programlisting>
  1168. </example>
  1169. <para>
  1170. Man kann den optionalen dritten Parameter von <methodname>quoteInto()</methodname>
  1171. verwenden um den <acronym>SQL</acronym> Datentyp zu spezifizieren. Nummerische
  1172. Datentypen werden nicht in Anführungszeichen gesetzt und andere Typen werden in
  1173. Anführungszeichen gesetzt.
  1174. </para>
  1175. <example id="zend.db.adapter.quoting.quote-into.example-2">
  1176. <title>Verwenden von quoteInto() mit einem SQL Typ</title>
  1177. <programlisting language="php"><![CDATA[
  1178. $sql = $db
  1179. ->quoteInto("SELECT * FROM bugs WHERE bug_id = ?", '1234', 'INTEGER');
  1180. echo $sql;
  1181. // SELECT * FROM bugs WHERE reported_by = 1234
  1182. ]]></programlisting>
  1183. </example>
  1184. </sect3>
  1185. <sect3 id="zend.db.adapter.quoting.quote-identifier">
  1186. <title>Nutzung von quoteIdentifier()</title>
  1187. <para>
  1188. Werte könnten nicht der einzige Teil der <acronym>SQL</acronym> Syntax sein, der
  1189. Variabel sein soll. Wenn <acronym>PHP</acronym> Variablen genutzt werden um
  1190. Tabellen, Spalten oder andere Bezeichner in den <acronym>SQL</acronym> Anweisungen
  1191. zu benennen, könnte es nötig sein das diese Strings ebenfalls quotiert werden
  1192. müssen. Standardmäßig haben <acronym>SQL</acronym> Bezeichner Syntaxregeln wie
  1193. <acronym>PHP</acronym> und die meißten anderen Programmiersprachen. Zum Beispiel
  1194. dürfen Bezeichner keine Leerzeichen, bestimmte Punktierung, spezielle Zeichen oder
  1195. Internationale Zeichen enthalten. Außerdem sind bestimmte Wörter für die
  1196. <acronym>SQL</acronym> Syntax reserviert und dürfen nicht als Bezeichner verwendet
  1197. werden.
  1198. </para>
  1199. <para>
  1200. Dennoch hat <acronym>SQL</acronym> ein Feature mit Namen
  1201. <emphasis>delimited identifiers (begrenzte Bezeichner)</emphasis>, welches eine
  1202. größere Auswahl bei der Schreibweise von Bezeichnern erlaubt. Wenn ein
  1203. <acronym>SQL</acronym> Bezeichner mit dem richtigen Typ von´ Quotierung
  1204. eingeschlossen ist, können Schreibweisen für die Bezeichner verwendet werden, die
  1205. ohne der Quotierung ungültig wären. Begrenzte Bezeichner können Leerzeichen,
  1206. Punktierung oder internationale Zeichen enthalten. Desweiteren dürfen auch von der
  1207. <acronym>SQL</acronym> Syntax reservierte Wörter verwendet werden, wenn sie von
  1208. Bezeichner Begrenzungszeichen eingeschlossen sind.
  1209. </para>
  1210. <para>
  1211. Die <methodname>quoteIdentifier()</methodname> Methode funktioniert wie
  1212. <methodname>quote()</methodname>, aber sie wendet die Bezeichner Begrenzungszeichen
  1213. entsprechend dem verwendeten Adapter an. Zum Beispiel nutzt Standard
  1214. <acronym>SQL</acronym> doppelte Anführungszeichen (") zum begrenzen von
  1215. Bezeichnern und die meisten der <acronym>RDBMS</acronym> Marken nutzen ebenfalls
  1216. dieses Symbol. MySQL hingegen benutzt back-quotes (`) als
  1217. Standardzeichen. Die <methodname>quoteIdentifier()</methodname> Methode ersetzt
  1218. außerdem spezielle Zeichen im String Argument.
  1219. </para>
  1220. <example id="zend.db.adapter.quoting.quote-identifier.example">
  1221. <title>Nutzung von quoteIdentifier()</title>
  1222. <programlisting language="php"><![CDATA[
  1223. // Wir könnten einen Tabellennamen haben, der ein in SQL reserviertes Wort ist
  1224. $tableName = $db->quoteIdentifier("order");
  1225. $sql = "SELECT * FROM $tableName";
  1226. echo $sql
  1227. // SELECT * FROM "order"
  1228. ]]></programlisting>
  1229. </example>
  1230. <para>
  1231. <acronym>SQL</acronym> begrenzte Bezeichner beachten die Groß- und Kleinschreibung,
  1232. im Gegensatz zu nicht quotierten Bezeichnern. Daher muss, bei Verwendung von
  1233. begrenztern Bezeichnern, die Schreibung der Bezeichner genau der Schreibung der
  1234. Bezeichner im Tabellenschema entsprechen. Einschließlich der Groß- und
  1235. Kleinschreibung.
  1236. </para>
  1237. <para>
  1238. In den meisten Fällen wo <acronym>SQL</acronym> innerhalb der
  1239. <classname>Zend_Db</classname> Klassen generiert wird, werden standardmäßig alle
  1240. Bezeichner automatisch begrenzt. Dieses Verhalten kann mit der Option
  1241. <constant>Zend_Db::AUTO_QUOTE_IDENTIFIERS</constant> geändert werden. Dies muss
  1242. beim Instanziieren des Adapters wie in <xref
  1243. linkend="zend.db.adapter.connecting.parameters.example2" /> angegeben werden.
  1244. </para>
  1245. </sect3>
  1246. </sect2>
  1247. <sect2 id="zend.db.adapter.transactions">
  1248. <title>Kontrollieren von Datenbank Transaktionen</title>
  1249. <para>
  1250. Datenbanken definieren Transaktionen als logische Einheiten von Arbeit, die als einzelne
  1251. Änderung übergeben oder rückgängig gemacht werden kann, selbst wenn sie auf
  1252. verschiedenen Tabellen operiert. Alle Anfragen an einen Datenbank werden im Kontext
  1253. einer Transaktion ausgeführt, selbst wenn der Datenbanktreiber sie implizit Verwaltet.
  1254. Es wird <emphasis>auto-commit</emphasis> Modus genannt, wenn der Datenbanktreiber eine
  1255. Transaktion für jede Anweisung erzeugt, und diese direkt nach dem Ausführen des
  1256. <acronym>SQL</acronym> Statements übergibt. Standardmäßig operieren alle
  1257. <classname>Zend_Db</classname> Adapterklassen im auto-commit Modus.
  1258. </para>
  1259. <para>
  1260. Alternativ kann der Begin und das Ergebnis einer Transaktion selbst spezifiziert
  1261. werden, und damit kann kontrolliert werden wieviele <acronym>SQL</acronym> Anfragen in
  1262. einer Gruppe enthalten sind, die entweder übergeben oder rückgängig gemacht wird, als
  1263. eine einzelne Operation. Um eine Transaktion zu initiieren wird die
  1264. <methodname>beginTransaction()</methodname> Methode verwendet. Anschließend folgende
  1265. <acronym>SQL</acronym> Anweisungen werden im Kontext der selben Transaktion ausgeführt
  1266. bis sie explizit aufgelöst wird.
  1267. </para>
  1268. <para>
  1269. Um eine Transaktion aufzulösen wird entweder die <methodname>commit()</methodname> oder
  1270. die <methodname>rollBack()</methodname> Methode verwendet. Die
  1271. <methodname>commit()</methodname> Methode markiert die Änderungen die während der
  1272. Transaktionen durchgeführt wurden als übergeben, was bedeutet das die Effekte dieser
  1273. Änderungen in anderen Transaktionen angezeigt werden.
  1274. </para>
  1275. <para>
  1276. Die <methodname>rollBack()</methodname> Methode tut das Gegenteil: sie verwirft die
  1277. Änderungen die während der Transaktionen durchgeführt wurden. Die Änderungen werden
  1278. gewissermaßen ungeschehen gemacht, der Status der Daten ändert sich zurück auf jenen
  1279. wie sie vor Beginn der Transaktion waren. Allerdings hat das rückgängig machen keinen
  1280. Einfluss auf Änderungen die von anderen, gleichzeitig laufenden Transaktionen
  1281. verursacht wurden.
  1282. </para>
  1283. <para>
  1284. Nach dem Auflösen der Transaktion befindet sich der
  1285. <classname>Zend_Db_Adapter</classname> wieder im auto-commit Modus, bis
  1286. <methodname>beginTransaction()</methodname> wieder aufgerufen wird.
  1287. </para>
  1288. <example id="zend.db.adapter.transactions.example">
  1289. <title>Verwalten einer Transaktion um Konsistenz sicher zu stellen</title>
  1290. <programlisting language="php"><![CDATA[
  1291. // Eine Transaktion explizit starten
  1292. $db->beginTransaction();
  1293. try {
  1294. // Versuchen einen oder mehrere Abfragen auszuführen
  1295. $db->query(...);
  1296. $db->query(...);
  1297. $db->query(...);
  1298. // Wenn alle erfolgreich waren, übertrage die Transaktion
  1299. // und alle Änderungen werden auf einmal übermittelt
  1300. $db->commit();
  1301. } catch (Exception $e) {
  1302. // Wenn irgendeine der Abfragen fehlgeschlagen ist, wirf eine Ausnahme, wir
  1303. // wollen die komplette Transaktion zurücknehmen, alle durch die
  1304. // Transaktion gemachten Änderungen wieder entfernen, auch die erfolgreichen
  1305. // So werden alle Änderungen auf einmal übermittelt oder keine
  1306. $db->rollBack();
  1307. echo $e->getMessage();
  1308. }
  1309. ]]></programlisting>
  1310. </example>
  1311. </sect2>
  1312. <sect2 id="zend.db.adapter.list-describe">
  1313. <title>Auflistung und Beschreibung von Tabellen</title>
  1314. <para>
  1315. Die <methodname>listTables()</methodname> Methode gibt ein Array von Strings zurück,
  1316. mit den Namen aller Tabellen in der aktuellen Datenbank.
  1317. </para>
  1318. <para>
  1319. Die <methodname>describeTable()</methodname> Methode gibt ein assoziatives Array von
  1320. MetaDaten über die Tabelle zurück. Das erste Argument dieser Methode ist ein String der
  1321. den Namen der Tabelle enthält. Das zweite Argument ist optional und benennt das Schema
  1322. in dem die Tabelle besteht.
  1323. </para>
  1324. <para>
  1325. Die Schlüssel des assoziativen Arrays sind die Spaltennamen der Tabelle. Der zugehörige
  1326. Wert jeder Spalte ist ebenfalls ein assoziatives Array mit den folgenden Schlüsseln und
  1327. Werten:
  1328. </para>
  1329. <table frame="all" cellpadding="5" id="zend.db.adapter.list-describe.metadata">
  1330. <title>Metadata Felder die von describeTable() zurückgegeben werden</title>
  1331. <tgroup cols="3" align="left" colsep="1" rowsep="1">
  1332. <thead>
  1333. <row>
  1334. <entry>Schlüssel</entry>
  1335. <entry>Typ</entry>
  1336. <entry>Beschreibung</entry>
  1337. </row>
  1338. </thead>
  1339. <tbody>
  1340. <row>
  1341. <entry><constant>SCHEMA_NAME</constant></entry>
  1342. <entry>(string)</entry>
  1343. <entry>Name des Datenbankschemas in welchem diese Tabelle existiert.</entry>
  1344. </row>
  1345. <row>
  1346. <entry><constant>TABLE_NAME</constant></entry>
  1347. <entry>(string)</entry>
  1348. <entry>Name der Tabelle zu welcher diese Spalte gehört.</entry>
  1349. </row>
  1350. <row>
  1351. <entry><constant>COLUMN_NAME</constant></entry>
  1352. <entry>(string)</entry>
  1353. <entry>Name der Spalte.</entry>
  1354. </row>
  1355. <row>
  1356. <entry><constant>COLUMN_POSITION</constant></entry>
  1357. <entry>(integer)</entry>
  1358. <entry>Ordinale Position der Spalte in der Tabelle.</entry>
  1359. </row>
  1360. <row>
  1361. <entry><constant>DATA_TYPE</constant></entry>
  1362. <entry>(string)</entry>
  1363. <entry>RDBMS Name des Datentyps der Spalte.</entry>
  1364. </row>
  1365. <row>
  1366. <entry><constant>DEFAULT</constant></entry>
  1367. <entry>(string)</entry>
  1368. <entry>Standardwert der Spalte, wenn angegeben.</entry>
  1369. </row>
  1370. <row>
  1371. <entry><constant>NULLABLE</constant></entry>
  1372. <entry>(boolean)</entry>
  1373. <entry>
  1374. <constant>TRUE</constant> wenn die Spalte <acronym>SQL</acronym>
  1375. <constant>NULL</constant> akzeptiert, <constant>FALSE</constant> wenn
  1376. die Spalte eine <constant>NOT</constant> <constant>NULL</constant>
  1377. Bedingung hat.
  1378. </entry>
  1379. </row>
  1380. <row>
  1381. <entry><constant>LENGTH</constant></entry>
  1382. <entry>(integer)</entry>
  1383. <entry>
  1384. Länge oder Größe der Spalte wie vom <acronym>RDBMS</acronym> angegeben.
  1385. </entry>
  1386. </row>
  1387. <row>
  1388. <entry><constant>SCALE</constant></entry>
  1389. <entry>(integer)</entry>
  1390. <entry>
  1391. Scalar vom Typ <acronym>SQL</acronym> NUMERIC oder
  1392. <constant>DECIMAL</constant>.
  1393. </entry>
  1394. </row>
  1395. <row>
  1396. <entry><constant>PRECISION</constant></entry>
  1397. <entry>(integer)</entry>
  1398. <entry>
  1399. Präzision des Typs <acronym>SQL</acronym> NUMERIC oder
  1400. <constant>DECIMAL</constant>.
  1401. </entry>
  1402. </row>
  1403. <row>
  1404. <entry><constant>UNSIGNED</constant></entry>
  1405. <entry>(boolean)</entry>
  1406. <entry>
  1407. <constant>TRUE</constant> wenn ein Integer-basierender Typ als
  1408. <constant>UNSIGNED</constant> angegeben wird.
  1409. </entry>
  1410. </row>
  1411. <row>
  1412. <entry><constant>PRIMARY</constant></entry>
  1413. <entry>(boolean)</entry>
  1414. <entry>
  1415. <constant>TRUE</constant> wenn die Spalte Teil des Primärschlüsssels der
  1416. Tabelle ist.
  1417. </entry>
  1418. </row>
  1419. <row>
  1420. <entry><constant>PRIMARY_POSITION</constant></entry>
  1421. <entry>(integer)</entry>
  1422. <entry>
  1423. Ordinale Position (1-basierend) der Spalte des Primärschlüssels.
  1424. </entry>
  1425. </row>
  1426. <row>
  1427. <entry><constant>IDENTITY</constant></entry>
  1428. <entry>(boolean)</entry>
  1429. <entry>
  1430. <constant>TRUE</constant> wenn die Spalte einen auto-increment Wert
  1431. nutzt.
  1432. </entry>
  1433. </row>
  1434. </tbody>
  1435. </tgroup>
  1436. </table>
  1437. <note>
  1438. <title>Wie das IDENTITY Metadata Feld zu speziellen RDBMS zuzuordnen ist</title>
  1439. <para>
  1440. Das IDENTITY Metadata Feld wurd gewählt als ein 'idiomatischer' Ausdruck um eine
  1441. Relation von Ersatzschlüsseln zu repräsentieren. Dieses Feld ist üblicherweise durch
  1442. die folgenden Werte bekannt:
  1443. </para>
  1444. <itemizedlist>
  1445. <listitem>
  1446. <para>
  1447. <constant>IDENTITY</constant> - DB2, MSSQL
  1448. </para>
  1449. </listitem>
  1450. <listitem>
  1451. <para>
  1452. <constant>AUTO_INCREMENT</constant> - MySQL
  1453. </para>
  1454. </listitem>
  1455. <listitem>
  1456. <para>
  1457. <constant>SERIAL</constant> - PostgreSQL
  1458. </para>
  1459. </listitem>
  1460. <listitem>
  1461. <para>
  1462. <constant>SEQUENCE</constant> - Oracle
  1463. </para>
  1464. </listitem>
  1465. </itemizedlist>
  1466. </note>
  1467. <para>
  1468. Wenn keine Tabelle mit dem Tabellennamen und dem optional angegebenen Schemanamen
  1469. existiert, gibt <methodname>describeTable()</methodname> ein leeres Array zurück.
  1470. </para>
  1471. </sect2>
  1472. <sect2 id="zend.db.adapter.closing">
  1473. <title>Schließen einer Verbindung</title>
  1474. <para>
  1475. Normalerweise ist es nicht nötig eine Datenbankverbindung zu schließen.
  1476. <acronym>PHP</acronym> räumt automatisch alle Ressourcen am Ende einer Anfrage auf und
  1477. die Datenbankerweiterungen sind so designed das sie Verbindungen beenden wenn Referenzen
  1478. zu ihren Objekten aufgeräumt werden.
  1479. </para>
  1480. <para>
  1481. Trotzdem könnte es sinnvoll sein, wenn ein lang andauerndes <acronym>PHP</acronym>
  1482. Script verwendet wird, das viele Datenbankverbindungen hat, diese zu schließen um zu
  1483. vermeiden das die Kapazität des <acronym>RDBMS</acronym> Servers überschritten wird. Die
  1484. <methodname>closeConnection()</methodname> Methode der Adapterklasse kann verwendet
  1485. werden um die zugrundeliegende Datenbankverbindung explizit zu schließen.
  1486. </para>
  1487. <para>
  1488. Seit Release 1.7.2, kann man prüfen ob man mit der
  1489. <methodname>isConnected()</methodname> prüfen ob man aktuell mit dem
  1490. <acronym>RDBMS</acronym> Server verbunden ist. Das bedeutet das eine Verbindungs
  1491. Ressource initialisiert und nicht geschlossen wurde. Diese Funktion ist aktuell nicht in
  1492. der Lage zu prüfen ob zum Beispiel die Server Seite die Verbindung geschlossen hat. Das
  1493. wird intern verwendet um die Verbindung zu schließen. Das erlaubt es die Verbindung ohne
  1494. Fehler mehrere Male zu schließen. Das war bereits vor 1.7.2 der Fall für
  1495. <acronym>PDO</acronym> Adapter, aber nicht für die anderen.
  1496. </para>
  1497. <example id="zend.db.adapter.closing.example">
  1498. <title>Schließen einer Datenbankverbindung</title>
  1499. <programlisting language="php"><![CDATA[
  1500. $db->closeConnection();
  1501. ]]></programlisting>
  1502. </example>
  1503. <note>
  1504. <title>Unterstützt Zend_Db persistente Verbindungen?</title>
  1505. <para>
  1506. Ja, Persistenz wird durch das Hinzufügen des <property>persistent</property> Flags
  1507. in der Konfiguration (nicht driver_configuration) und dessen Setzen auf
  1508. <constant>TRUE</constant> bei einem Adapter in <classname>Zend_Db</classname>
  1509. unterstützt.
  1510. </para>
  1511. <example id="zend.db.adapter.connecting.persistence.example">
  1512. <title>Verwendung des Persistence Flags mit dem Oracle Adapter</title>
  1513. <programlisting language="php"><![CDATA[
  1514. $db = Zend_Db::factory('Oracle', array(
  1515. 'host' => '127.0.0.1',
  1516. 'username' => 'webuser',
  1517. 'password' => 'xxxxxxxx',
  1518. 'dbname' => 'test',
  1519. 'persistent' => true
  1520. ));
  1521. ]]></programlisting>
  1522. </example>
  1523. <para>
  1524. Es ist zu beachten das die Verwendung von persistenten Verbindungen einen Exzess
  1525. an Idle Verbindungen auf dem <acronym>RDBMS</acronym> Server verursachen kann, was
  1526. mehr Probleme macht als jeder Performance Gewinn den man durch die Verminderung des
  1527. Overheads eines Verbindungsaufbaues erhalten kann.
  1528. </para>
  1529. <para>
  1530. Datenbankverbindungen haben einen Status. Natürlich existieren einige Objekte auf
  1531. dem <acronym>RDBMS</acronym> Server im Gültigkeitsbereich einer Session. Beispiele
  1532. dafür sind locks, user variablen, temporary tables und Informationen über die
  1533. zuletzt ausgeführte Anfrage, sowie betroffene Zeilen und zuletzt generierte ID
  1534. Werte. Wenn persistente Verbindungen genutzt werden könnte die Anwendung Zugriff auf
  1535. ungültige oder privilegierte Daten erlangen, die in einem vorigen
  1536. <acronym>PHP</acronym> Request erzeugt wurden.
  1537. </para>
  1538. <para>
  1539. Aktuell unterstützen nur die Oracle, DB2 und <acronym>PDO</acronym> Adapter (wo es
  1540. von <acronym>PHP</acronym> spezifiziert ist) Persistenz in
  1541. <classname>Zend_Db</classname>.
  1542. </para>
  1543. </note>
  1544. </sect2>
  1545. <sect2 id="zend.db.adapter.other-statements">
  1546. <title>Ausführen anderer Datenbank Anweisungen</title>
  1547. <para>
  1548. Es könnte Fälle geben in denen direkter Zugriff auf das Verbindungsobjekt benötigt wird,
  1549. wie es von der <acronym>PHP</acronym> Erweiterung bereitgestellt wird. Einige der
  1550. Erweiterungen könnten Features anbieten, welche nicht von Methoden der
  1551. <classname>Zend_Db_Adapter_Abstract</classname> Klasse auftauchen..
  1552. </para>
  1553. <para>
  1554. Zum Beispiel werden alle <acronym>SQL</acronym> Anweisungen von
  1555. <classname>Zend_Db</classname> vorbereitet und dann ausgeführt. Trotzdem gibt es einige
  1556. Features welche nicht kompatibel mit vorbereiteten Anweisungen sind. DDL Anweisungen wie
  1557. CREATE und ALTER können in MySQL nicht vorbereitet werden. Auch können
  1558. <acronym>SQL</acronym> Anweisungen keinen Nutzen aus dem <ulink
  1559. url="http://dev.mysql.com/doc/refman/5.1/en/query-cache-how.html">MySQL Query
  1560. Cache</ulink> ziehen, bei einer geringeren MySQL Version als 5.1.17.
  1561. </para>
  1562. <para>
  1563. Die meisten <acronym>PHP</acronym> Datenbankerweiterungen bieten eine Methode um
  1564. <acronym>SQL</acronym> Anweisung auszuführen ohne diese vorzubereiten. Zum Beispiel
  1565. bietet <acronym>PDO</acronym> die Methode <methodname>exec()</methodname>. Das
  1566. Verbindungsobjekt der <acronym>PHP</acronym> Erweiterung kann kann mit der Methode
  1567. <methodname>getConnection()</methodname> direkt verwendet werden.
  1568. </para>
  1569. <example id="zend.db.adapter.other-statements.example">
  1570. <title>Ausführen eines nicht-prepared Statements mit einem PDO Adapter</title>
  1571. <programlisting language="php"><![CDATA[
  1572. $result = $db->getConnection()->exec('DROP TABLE bugs');
  1573. ]]></programlisting>
  1574. </example>
  1575. <para>
  1576. So ähnlich können auch andere Methoden oder Eigenschaften der speziellen
  1577. <acronym>PHP</acronym> Datenbankerweiterung genutzt werden. Zu beachten dabei ist
  1578. jedoch, dass dadurch möglicherweise die Anwendung auf das angegebene Interface,
  1579. bereitgestellt von einer Erweiterung für ein bestimmtes <acronym>RDBMS</acronym>,
  1580. beschränkt wird.
  1581. </para>
  1582. <para>
  1583. In zukünftigen Versionen von <classname>Zend_Db</classname> werden Möglichkeiten gegeben
  1584. sein, um Methoden Startpunkte hinzuzufügen, für Funktionalitäten die den unterstützten
  1585. <acronym>PHP</acronym> Datenbankerweiterungen gemein sind. Dies wird die
  1586. Rückwärtskompatibilität nicht beeinträchtigen.
  1587. </para>
  1588. </sect2>
  1589. <sect2 id="zend.db.adapter.server-version">
  1590. <title>Erhalten der Server Version</title>
  1591. <para>
  1592. Seit Release 1.7.2 kann man die Version des Servers in einem <acronym>PHP</acronym>
  1593. artigen Stil erhalten damit man es mit <methodname>version_compare()</methodname>
  1594. verwenden kann. Wenn die Information nicht vorhanden ist erhält man
  1595. <constant>NULL</constant> zurück.
  1596. </para>
  1597. <example id="zend.db.adapter.server-version.example">
  1598. <title>Prüfen der Server Version bevor eine Abfrage gestartet wird</title>
  1599. <programlisting language="php"><![CDATA[
  1600. $version = $db->getServerVersion();
  1601. if (!is_null($version)) {
  1602. if (version_compare($version, '5.0.0', '>=')) {
  1603. // mach was
  1604. } else {
  1605. // mach was anderes
  1606. }
  1607. } else {
  1608. // Server Version ist unmöglich zu lesen
  1609. }
  1610. ]]></programlisting>
  1611. </example>
  1612. </sect2>
  1613. <sect2 id="zend.db.adapter.adapter-notes">
  1614. <title>Anmerkungen zu bestimmten Adaptern</title>
  1615. <para>
  1616. Dieser Abschnitt beschreibt Unterschiede zwischen den verschieden Adapterklassen auf die
  1617. man achtgeben sollte.
  1618. </para>
  1619. <sect3 id="zend.db.adapter.adapter-notes.sqlsrv">
  1620. <title>Microsoft SQL Server</title>
  1621. <itemizedlist>
  1622. <listitem>
  1623. <para>
  1624. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1625. dem Namen 'Sqlsrv' angegeben.
  1626. </para>
  1627. </listitem>
  1628. <listitem>
  1629. <para>
  1630. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterung sqlsrv.
  1631. </para>
  1632. </listitem>
  1633. <listitem>
  1634. <para>
  1635. Microsoft <acronym>SQL</acronym> Server unterstützt keine Sequenzen, daher
  1636. ignoriert <methodname>lastInsertId()</methodname> das Primary Key Argument
  1637. und gibt immer den letzten Wert zurück der für den auto-increment Schlüssel
  1638. generiert wurde wenn ein Tabellenname spezifiziert wurde oder die letzte
  1639. Insert Abfrage eine Id zurückgegeben hat. Die
  1640. <methodname>lastSequenceId()</methodname> Methode gibt
  1641. <constant>NULL</constant> zurück.
  1642. </para>
  1643. </listitem>
  1644. <listitem>
  1645. <para>
  1646. <classname>Zend_Db_Adapter_Sqlsrv</classname> setzt
  1647. <constant>QUOTED_IDENTIFIER ON</constant> unmittelbar nach der
  1648. Verbindung zu einer <acronym>SQL</acronym> Server Datenbank. Dadurch
  1649. verwendet der Treiber das standardmäßige <acronym>SQL</acronym> Trennzeichen
  1650. (<emphasis>"</emphasis>) statt den propietären eckigen Klammern die der
  1651. <acronym>SQL</acronym> Server für die Identifikatoren als Trennzeichen
  1652. verwendet.
  1653. </para>
  1654. </listitem>
  1655. <listitem>
  1656. <para>
  1657. Man kann <property>driver_options</property> als Schlüssel im Options Array
  1658. spezifizieren. Der Wert kann alles hieraus sein: <ulink
  1659. url="http://msdn.microsoft.com/en-us/library/cc296161(SQL.90).aspx">http://msdn.microsoft.com/en-us/library/cc296161(SQL.90).aspx</ulink>.
  1660. </para>
  1661. </listitem>
  1662. <listitem>
  1663. <para>
  1664. Man kann <methodname>setTransactionIsolationLevel()</methodname> verwenden
  1665. um einen Isolations Level für die aktuelle Verbindung zu setzen. Der Wert
  1666. kann wie folgt sein: <constant>SQLSRV_TXN_READ_UNCOMMITTED</constant>,
  1667. <constant>SQLSRV_TXN_READ_COMMITTED</constant>,
  1668. <constant>SQLSRV_TXN_REPEATABLE_READ</constant>,
  1669. <constant>SQLSRV_TXN_SNAPSHOT</constant> oder
  1670. <constant>SQLSRV_TXN_SERIALIZABLE</constant>.
  1671. </para>
  1672. </listitem>
  1673. <listitem>
  1674. <para>
  1675. Mit <acronym>ZF</acronym> 1.9 ist das mindestens unterstützte Build der
  1676. <acronym>PHP</acronym> <acronym>SQL</acronym> Server erweiterung von
  1677. Microsoft 1.0.1924.0 und die Version des <acronym>MSSQL</acronym> Server
  1678. Native Clients 9.00.3042.00.
  1679. </para>
  1680. </listitem>
  1681. </itemizedlist>
  1682. </sect3>
  1683. <sect3 id="zend.db.adapter.adapter-notes.ibm-db2">
  1684. <title>IBM DB2</title>
  1685. <itemizedlist>
  1686. <listitem>
  1687. <para>
  1688. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1689. dem Namen 'Db2' angegeben.
  1690. </para>
  1691. </listitem>
  1692. <listitem>
  1693. <para>
  1694. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterung ibm_db2.
  1695. </para>
  1696. </listitem>
  1697. <listitem>
  1698. <para>
  1699. IBM DB2 unterstützt sowohl Sequenzen als auch auto-increment Schlüssel.
  1700. Daher sind die Argumente für <methodname>lastInsertId()</methodname>
  1701. optional. Werden keine Argumente angegeben, gibt der Adapter den letzten
  1702. Wert der für den auto-increment Key generiert wurde zurück. Werden
  1703. Argumente angegeben, gibt der Adapter den letzten Wert der für die Sequenz
  1704. mit dem Namen, entsprechend der Konvention,
  1705. '<emphasis>table</emphasis>_<emphasis>column</emphasis>_seq' generiert
  1706. wurde zurück.
  1707. </para>
  1708. </listitem>
  1709. </itemizedlist>
  1710. </sect3>
  1711. <sect3 id="zend.db.adapter.adapter-notes.mysqli">
  1712. <title>MySQLi</title>
  1713. <itemizedlist>
  1714. <listitem>
  1715. <para>
  1716. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1717. dem Namen 'Mysqli' angegeben.
  1718. </para>
  1719. </listitem>
  1720. <listitem>
  1721. <para>
  1722. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterung mysqli.
  1723. </para>
  1724. </listitem>
  1725. <listitem>
  1726. <para>
  1727. MySQL unterstützt keine Sequenzen, daher ignoriert
  1728. <methodname>lastInsertId()</methodname> Argumente und gibt immer den
  1729. letzten Wert der für den auto-increment Schlüssel generiert wurde zurück.
  1730. Die <methodname>lastSequenceId()</methodname> Methode gibt
  1731. <constant>NULL</constant> zurück.
  1732. </para>
  1733. </listitem>
  1734. </itemizedlist>
  1735. </sect3>
  1736. <sect3 id="zend.db.adapter.adapter-notes.oracle">
  1737. <title>Oracle</title>
  1738. <itemizedlist>
  1739. <listitem>
  1740. <para>
  1741. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1742. dem Namen 'Oracle' angegeben.
  1743. </para>
  1744. </listitem>
  1745. <listitem>
  1746. <para>
  1747. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterung oci8.
  1748. </para>
  1749. </listitem>
  1750. <listitem>
  1751. <para>
  1752. Oracle unterstützt keine auto-increment Schlüssel, daher sollte der Name
  1753. einer Sequenz an <methodname>lastInsertId()</methodname> oder
  1754. <methodname>lastSequenceId()</methodname> übergeben werden.
  1755. </para>
  1756. </listitem>
  1757. <listitem>
  1758. <para>
  1759. Die Oracle Erweiterung unterstützt keine positionierten Parameter. Es müssen
  1760. benannte Parameter verwendet werden.
  1761. </para>
  1762. </listitem>
  1763. <listitem>
  1764. <para>
  1765. Aktuell wird die <constant>Zend_Db::CASE_FOLDING</constant> Option vom
  1766. Oracle Adapter nicht unterstützt. Um diese Option mit Oracle zu nutzen muss
  1767. der <acronym>PDO</acronym> OCI Adapter verwendet werden.
  1768. </para>
  1769. </listitem>
  1770. <listitem>
  1771. <para>
  1772. Standardmäßig werden LOB Felder als OCI-Log Objekte zurückgegeben. Man kann
  1773. Sie für alle Anfragen als String empfangen indem die Treiberoption
  1774. '<property>lob_as_string</property>' verwendet wird, oder für spezielle Anfragen
  1775. durch Verwendung von <methodname>setLobAsString(boolean)</methodname> auf
  1776. dem Adapter oder dem Statement.
  1777. </para>
  1778. </listitem>
  1779. </itemizedlist>
  1780. </sect3>
  1781. <sect3 id="zend.db.adapter.adapter-notes.pdo-ibm">
  1782. <title>PDO Adapter für IBM DB2 und für Informix Dynamic Server (IDS)</title>
  1783. <itemizedlist>
  1784. <listitem>
  1785. <para>
  1786. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1787. dem Namen '<classname>Pdo_Ibm</classname>' spezifiziert.
  1788. </para>
  1789. </listitem>
  1790. <listitem>
  1791. <para>
  1792. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterungen pdo und
  1793. pdo_ibm.
  1794. </para>
  1795. </listitem>
  1796. <listitem>
  1797. <para>
  1798. Es muß mindestens die PDO_IBM Erweiterung mit der Version
  1799. 1.2.2 verwendet werden. Wenn eine ältere Version verwendet wird, muß die
  1800. PDO_IBM Erweiterung über <acronym>PECL</acronym>
  1801. hochgerüstet werden.
  1802. </para>
  1803. </listitem>
  1804. </itemizedlist>
  1805. </sect3>
  1806. <sect3 id="zend.db.adapter.adapter-notes.pdo-mssql">
  1807. <title>PDO Microsoft SQL Server</title>
  1808. <itemizedlist>
  1809. <listitem>
  1810. <para>
  1811. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1812. dem Namen '<classname>Pdo_Mssql</classname>' angegeben.
  1813. </para>
  1814. </listitem>
  1815. <listitem>
  1816. <para>
  1817. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterungen pdo und
  1818. pdo_dblib.
  1819. </para>
  1820. </listitem>
  1821. <listitem>
  1822. <para>
  1823. Microsoft <acronym>SQL</acronym> Server unterstützt keine Sequenzen, daher
  1824. ignoriert <methodname>lastInsertId()</methodname> Argumente und gibt immer
  1825. den letzten Wert der für den auto-increment Schlüssel generiert wurde
  1826. zurück. Die <methodname>lastSequenceId()</methodname> Methode gibt
  1827. <constant>NULL</constant> zurück.
  1828. </para>
  1829. </listitem>
  1830. <listitem>
  1831. <para>
  1832. Wenn man mit Unicode Strings in einer anderen Codierung als UCS-2 (wie
  1833. UTF-8) arbeitet, kann es sein das man eine Konvertierung im eigenen
  1834. Anwendungscode durchführen, oder die Daten in einer Binären Spalte speichern
  1835. muß. Referieren Sie bitte auf
  1836. <ulink url="http://support.microsoft.com/kb/232580">Microsoft's Knowledge
  1837. Base</ulink> für weitere Informationen.
  1838. </para>
  1839. </listitem>
  1840. <listitem>
  1841. <para>
  1842. <classname>Zend_Db_Adapter_Pdo_Mssql</classname> setzt
  1843. <constant>QUOTED_IDENTIFIER ON</constant> direkt nach dem Verbinden zu
  1844. einer <acronym>SQL</acronym> Server Datenbank. Dadurch verwendet der Treiber
  1845. das Standard <acronym>SQL</acronym> Bezeichner Begrenzungssymbol
  1846. (") an Stelle der proprietären Eckige-Klammer Syntax die der
  1847. <acronym>SQL</acronym> Server standradmäßig nutzt.
  1848. </para>
  1849. </listitem>
  1850. <listitem>
  1851. <para>
  1852. Es kann <property>pdoType</property> als ein Schlüssel im Optionsarray
  1853. gesetzt werden. Der Wert kann "mssql" (der Standard), "dblib", "freetds",
  1854. oder "sybase" sein. Diese Option beeinflusst den <acronym>DNS</acronym>
  1855. Prefix, welchen der Adapter beim Konstruieren des <acronym>DNS</acronym>
  1856. Strings benutzt. Sowohl "freetds" als auch "sybase" implementieren einen
  1857. Prefix von "sybase:", welcher für den <ulink
  1858. url="http://www.freetds.org/">FreeTDS</ulink> Satz von Libraries
  1859. verwendet wird. Siehe auch <ulink
  1860. url="http://www.php.net/manual/de/ref.pdo-dblib.connection.php">http://www.php.net/manual/de/ref.pdo-dblib.connection.php</ulink>
  1861. für weitere Informationen über die <acronym>DNS</acronym> Prefixe die von
  1862. diesem Treiber verwendet werden.
  1863. </para>
  1864. </listitem>
  1865. </itemizedlist>
  1866. </sect3>
  1867. <sect3 id="zend.db.adapter.adapter-notes.pdo-mysql">
  1868. <title>PDO MySQL</title>
  1869. <itemizedlist>
  1870. <listitem>
  1871. <para>
  1872. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1873. dem Namen '<classname>Pdo_Mysql</classname>' angegeben.
  1874. </para>
  1875. </listitem>
  1876. <listitem>
  1877. <para>
  1878. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterungen pdo und
  1879. pdo_mysql.
  1880. </para>
  1881. </listitem>
  1882. <listitem>
  1883. <para>
  1884. MySQL unterstützt keine Sequenzen, daher ignoriert
  1885. <methodname>lastInsertId()</methodname> Argumente und gibt immer den
  1886. letzten Wert der für den auto-increment Schlüssel generiert wurde zurück.
  1887. Die <methodname>lastSequenceId()</methodname> Methode gibt
  1888. <constant>NULL</constant> zurück.
  1889. </para>
  1890. </listitem>
  1891. </itemizedlist>
  1892. </sect3>
  1893. <sect3 id="zend.db.adapter.adapter-notes.pdo-oci">
  1894. <title>PDO Oracle</title>
  1895. <itemizedlist>
  1896. <listitem>
  1897. <para>
  1898. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1899. dem Namen '<classname>Pdo_Oci</classname>' angegeben.
  1900. </para>
  1901. </listitem>
  1902. <listitem>
  1903. <para>
  1904. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterungen pdo und
  1905. pdo_oci.
  1906. </para>
  1907. </listitem>
  1908. <listitem>
  1909. <para>
  1910. Oracle unterstützt keine auto-increment Schlüssel, daher sollte der Name
  1911. einer Sequenz an <methodname>lastInsertId()</methodname> oder
  1912. <methodname>lastSequenceId()</methodname> übergeben werden.
  1913. </para>
  1914. </listitem>
  1915. </itemizedlist>
  1916. </sect3>
  1917. <sect3 id="zend.db.adapter.adapter-notes.pdo-pgsql">
  1918. <title>PDO PostgreSQL</title>
  1919. <itemizedlist>
  1920. <listitem>
  1921. <para>
  1922. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1923. dem Namen '<classname>Pdo_Pgsql</classname>' angegeben.
  1924. </para>
  1925. </listitem>
  1926. <listitem>
  1927. <para>
  1928. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterungen pdo und
  1929. pdo_pgsql.
  1930. </para>
  1931. </listitem>
  1932. <listitem>
  1933. <para>
  1934. PostgreSQL unterstützt sowohl Sequenzen als auch auto-increment Schlüssel.
  1935. Daher sind die Argumente für <methodname>lastInsertId()</methodname>
  1936. optional. Werden keine Argumente angegeben, gibt der Adapter den letzten
  1937. Wert der für den auto-increment Key generiert wurde zurück. Werden
  1938. Argumente angegeben, gibt der Adapter den letzten Wert der für die Sequenz
  1939. mit dem Namen, entsprechend der Konvention,
  1940. '<emphasis>table</emphasis>_<emphasis>column</emphasis>_seq' generiert
  1941. wurde zurück.
  1942. </para>
  1943. </listitem>
  1944. </itemizedlist>
  1945. </sect3>
  1946. <sect3 id="zend.db.adapter.adapter-notes.pdo-sqlite">
  1947. <title>PDO SQLite</title>
  1948. <itemizedlist>
  1949. <listitem>
  1950. <para>
  1951. Dieser Adapter wird in der <methodname>factory()</methodname> Methode mit
  1952. dem Namen '<classname>Pdo_Sqlite</classname>' angegeben.
  1953. </para>
  1954. </listitem>
  1955. <listitem>
  1956. <para>
  1957. Dieser Adapter nutzt die <acronym>PHP</acronym> Erweiterungen pdo und
  1958. pdo_sqlite.
  1959. </para>
  1960. </listitem>
  1961. <listitem>
  1962. <para>
  1963. SQLite unterstützt keine Sequenzen, daher ignoriert
  1964. <methodname>lastInsertId()</methodname> Argumente und gibt immer den
  1965. letzten Wert der für den auto-increment Schlüssel generiert wurde zurück.
  1966. Die <methodname>lastSequenceId()</methodname> Methode gibt
  1967. <constant>NULL</constant> zurück.
  1968. </para>
  1969. </listitem>
  1970. <listitem>
  1971. <para>
  1972. Um mit einer SQLite2 Datenbank zu Verbinden muss
  1973. <command>'sqlite2' => true</command> in dem Array von Parametern beim
  1974. Erzeugen einer Instanz des <classname>Pdo_Sqlite</classname> Adapters
  1975. angegeben werden.
  1976. </para>
  1977. </listitem>
  1978. <listitem>
  1979. <para>
  1980. Um mit einer in-memory SQLite Datenbank zu verbinden muss
  1981. <command>'dbname' => ':memory:'</command> in dem Array von Parametern beim
  1982. Erzeugen einer Instanz des <classname>Pdo_Sqlite</classname> Adapters
  1983. angegeben werden.
  1984. </para>
  1985. </listitem>
  1986. <listitem>
  1987. <para>
  1988. Ältere Versionen des SQLite Treibers in <acronym>PHP</acronym> scheinen die
  1989. PRAGMA Kommandos nicht zu unterstützen, die benötigt werden um
  1990. sicherzustellen, dass kurze Spaltennamen in Ergebnissätzen verwendet werden.
  1991. Wenn in den Ergebnissätzen Schlüssel der Art "tabellenname.spaltenname" bei
  1992. Nutzung von JOIN Abfragen auftreten, sollte die aktuellste
  1993. <acronym>PHP</acronym> Version installiert werden.
  1994. </para>
  1995. </listitem>
  1996. </itemizedlist>
  1997. </sect3>
  1998. <sect3 id="zend.db.adapter.adapter-notes.firebird">
  1999. <title>Firebird/Interbase</title>
  2000. <itemizedlist>
  2001. <listitem>
  2002. <para>
  2003. Dieser Adapter verwendet die <acronym>PHP</acronym> Erweiterung
  2004. php_interbase.
  2005. </para>
  2006. </listitem>
  2007. <listitem>
  2008. <para>
  2009. Firebird/interbase unterstützt keine auto-increment Schlüssel, deswegen
  2010. sollte der Name einer Sequenz bei <methodname>lastInsertId()</methodname>
  2011. oder <methodname>lastSequenceId()</methodname> spezifiziert werden.
  2012. </para>
  2013. </listitem>
  2014. <listitem>
  2015. <para>
  2016. Aktuell wird die <constant>Zend_Db::CASE_FOLDING</constant> Option vom
  2017. Firebird/interbase Adapter nicht unterstützt. Nicht gequotete Identifizierer
  2018. werden automatisch in Großschreibweise zurückgegeben.
  2019. </para>
  2020. </listitem>
  2021. <listitem>
  2022. <para>
  2023. Der Name des Adapters ist <classname>ZendX_Db_Adapter_Firebird</classname>.
  2024. </para>
  2025. <para>
  2026. Beachte das der Parameter adapterNamespace mit dem Wert
  2027. <classname>ZendX_Db_Adapter</classname> zu verwenden ist.
  2028. </para>
  2029. <para>
  2030. Wir empfehlen die <filename>gds32.dll</filename> (oder Ihr Linux Äquivalent)
  2031. welche mit PHP ausgeliefert wird, auf die gleiche Version wie am Server
  2032. hochzurüsten. Für Firebird ist das Äquivalent zu
  2033. <filename>gds32.dll</filename> die <filename>fbclient.dll</filename>.
  2034. </para>
  2035. <para>
  2036. Standardmäßig werden alle Identifikatoren (Tabellennamen, Felder) in
  2037. Großschreibweise zurückgegeben.
  2038. </para>
  2039. </listitem>
  2040. </itemizedlist>
  2041. </sect3>
  2042. </sect2>
  2043. </sect1>
  2044. <!--
  2045. vim:se ts=4 sw=4 et:
  2046. -->