Zend_Db_Select.xml 63 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- EN-Revision: 24249 -->
  3. <!-- Reviewed: 21584 -->
  4. <sect1 id="zend.db.select">
  5. <title>Zend_Db_Select</title>
  6. <sect2 id="zend.db.select.introduction">
  7. <title>Einführung</title>
  8. <para>
  9. Das <classname>Zend_Db_Select</classname>-Objekt repräsentiert ein
  10. <acronym>SQL</acronym>-<acronym>SELECT</acronym>-Anfrage-Statement. Die Klasse bietet
  11. Methoden um einzelne Teile der Anfrage hinzuzufügen. Einzelne Teile der Anfrage können
  12. mit Hilfe von <acronym>PHP</acronym> Methoden und Datenstrukturen angegeben werden, und
  13. die Klasse erzeugt korrekte <acronym>SQL</acronym>-Syntax. Nachdem die Anfrage
  14. formuliert wurde, kann sie ausgeführt werden, als wäre sie mit einem normalen String
  15. geschrieben worden.
  16. </para>
  17. <para>
  18. <classname>Zend_Db_Select</classname> bietet folgenden Nutzen:
  19. </para>
  20. <itemizedlist>
  21. <listitem>
  22. <para>
  23. Objektorientierte Methoden um <acronym>SQL</acronym>-Anfragen Stück für Stück
  24. zu formulieren.
  25. </para>
  26. </listitem>
  27. <listitem>
  28. <para>
  29. Datenbankunabhängige Abstraktion einiger Teile der
  30. <acronym>SQL</acronym>-Anfrage.
  31. </para>
  32. </listitem>
  33. <listitem>
  34. <para>
  35. In den meisten Fällen automatische Quotierung von Metadaten um zu
  36. erlauben, dass Bezeichner reservierte <acronym>SQL</acronym>-Wörter und
  37. spezielle Zeichen enthalten dürfen.
  38. </para>
  39. </listitem>
  40. <listitem>
  41. <para>
  42. Quotierung von Bezeichnern und Werten, um das Risiko von
  43. Angriffen durch <acronym>SQL</acronym>-Injektion zu verringern.
  44. </para>
  45. </listitem>
  46. </itemizedlist>
  47. <para>
  48. Nutzung von <classname>Zend_Db_Select</classname> ist nicht zwingend erforderlich. Für
  49. einfache <acronym>SELECT</acronym>-Anfragen ist es normalerweise einfacher, die gesamte
  50. <acronym>SQL</acronym>-Anfrage in einem String zu formulieren und mit Hilfe der Methoden
  51. der Adapterklasse, wie <methodname>query()</methodname> oder
  52. <methodname>fetchAll()</methodname>, auszuführen. Die Nutzung von
  53. <classname>Zend_Db_Select</classname> ist hilfreich, wenn eine
  54. <acronym>SELECT</acronym>-Anfrage prozedural oder basierend auf der konditionellen
  55. Logik der Anwendung zusammengesetzt wird.
  56. </para>
  57. </sect2>
  58. <sect2 id="zend.db.select.creating">
  59. <title>Erzeugung eines Select-Objekts</title>
  60. <para>
  61. Die Instanz eines <classname>Zend_Db_Select</classname>-Objekts kann mit Hilfe der
  62. Methode <methodname>select()</methodname> des
  63. <classname>Zend_Db_Adapter_Abstract</classname>-Objekts erzeugt werden.
  64. </para>
  65. <example id="zend.db.select.creating.example-db">
  66. <title>Beispiel für die Nutzung der Methode select() der Datenbankadapterklasse</title>
  67. <programlisting language="php"><![CDATA[
  68. $db = Zend_Db::factory( ...Optionen... );
  69. $select = $db->select();
  70. ]]></programlisting>
  71. </example>
  72. <para>
  73. Ein anderer Weg ein <classname>Zend_Db_Select</classname>-Objekt zu erzeugen, ist die
  74. Nutzung des Konstruktors unter Angabe des Datenbankadapters als Argument.
  75. </para>
  76. <example id="zend.db.select.creating.example-new">
  77. <title>Beispiel für die Erzeugung eines Select-Objekts</title>
  78. <programlisting language="php"><![CDATA[
  79. $db = Zend_Db::factory( ...Optionen... );
  80. $select = new Zend_Db_Select($db);
  81. ]]></programlisting>
  82. </example>
  83. </sect2>
  84. <sect2 id="zend.db.select.building">
  85. <title>Erstellung von Select-Anfragen</title>
  86. <para>
  87. Wenn die Anfrage erstellt wird, können Bedingungen der Anfrage nacheinander hinzugefügt
  88. werden. Es gibt separate Methoden für das Hinzufügen von verschiedenen Bedingungen
  89. zum <classname>Zend_Db_Select</classname>-Objekt.
  90. </para>
  91. <example id="zend.db.select.building.example">
  92. <title>Beispiele für die Nutzung der Methoden zum Hinzufügen von Bedingungen</title>
  93. <programlisting language="php"><![CDATA[
  94. // Erzeugung des Zend_Db_Select-Objekts
  95. $select = $db->select();
  96. // Hinzufügen einer FROM Bedingung
  97. $select->from( ...Angabe von Tabelle und Spalten... )
  98. // Hinzufügen einer WHERE Bedingung
  99. $select->where( ...Angabe von Suchkriterien... )
  100. // Hinzufügen einer ORDER BY Bedingung
  101. $select->order( ...Angabe von Sortierkriterien... );
  102. ]]></programlisting>
  103. </example>
  104. <para>
  105. Die meisten Methoden des <classname>Zend_Db_Select</classname>-Objekts lassen sich auch
  106. über das bequeme Fluent Interface nutzen. Fluent Interface bedeutet, dass jede Methode
  107. eine Referenz auf das aufrufende Objekt zurück gibt, daher kann direkt eine andere
  108. Methode aufgerufen werden.
  109. </para>
  110. <example id="zend.db.select.building.example-fluent">
  111. <title>Beispiel für die Nutzung des Fluent Interface</title>
  112. <programlisting language="php"><![CDATA[
  113. $select = $db->select()
  114. ->from( ...Angabe von Tabelle und Spalten... )
  115. ->where( ...Angabe von Suchkriterien... )
  116. ->order( ...Angabe von Sortierkriterien... );
  117. ]]></programlisting>
  118. </example>
  119. <para>
  120. Die Beispiele in diesem Abschnitt zeigen die Nutzung des Fluent Interface, es kann aber
  121. auch immer das normale Interface verwendet werden. Häufig ist es nötig das normale
  122. Interface zu nutzen, zum Beispiel wenn die Anwendung vor dem Hinzufügen der Bedingung
  123. Berechnungen durchführen muss.
  124. </para>
  125. <sect3 id="zend.db.select.building.from">
  126. <title>Hinzufügen eines FROM Abschnitts</title>
  127. <para>
  128. Um die Tabelle für die Anfrage anzugeben, wird die Methode
  129. <methodname>from()</methodname>
  130. verwendet. Der Tabellenname kann als einfacher String übergeben werden.
  131. <classname>Zend_Db_Select</classname> wendet Quotierung auf Bezeichner an, es können
  132. also auch spezielle Zeichen verwendet werden.
  133. </para>
  134. <example id="zend.db.select.building.from.example">
  135. <title>Beispiel für die Methode from()</title>
  136. <programlisting language="php"><![CDATA[
  137. // Erstellen dieser Anfrage:
  138. // SELECT *
  139. // FROM "products"
  140. $select = $db->select()
  141. ->from('products');
  142. ]]></programlisting>
  143. </example>
  144. <para>
  145. Es kann auch der Beziehungsname (auch Aliasname genannt) einer Tabelle angegeben
  146. werden. Anstelle eines einfachen Strings muss dann ein assoziatives Array übergeben
  147. werden, welches den Beziehungsnamen dem tatsächlichen Tabellennamen zuordnet. In
  148. anderen Bedingungen der <acronym>SQL</acronym>-Anfrage kann dann dieser
  149. Beziehungsname verwendet werden. Wenn die Anfrage mehr als eine Tabelle verbindet,
  150. generiert <classname>Zend_Db_Select</classname> eindeutige Beziehungsnamen basierend
  151. auf den Tabellennamen, wenn keine Beziehungsnamen angegeben wurden.
  152. </para>
  153. <example id="zend.db.select.building.from.example-cname">
  154. <title>Beispiel für das Angeben eines Beziehungsnamens</title>
  155. <programlisting language="php"><![CDATA[
  156. // Erzeugt diese Anfrage:
  157. // SELECT p.*
  158. // FROM "products" AS p
  159. $select = $db->select()
  160. ->from(array('p' => 'products'));
  161. ]]></programlisting>
  162. </example>
  163. <para>
  164. Einige <acronym>RDBMS</acronym>-Marken unterstützen einen voran stehenden
  165. Schemenbezeichner für eine Tabelle. Der Tabellenname kann mit
  166. "<command>schemaName.tabellenName</command>" angegeben werden,
  167. <classname>Zend_Db_Select</classname> quotiert die einzelnen Teile für sich. Der
  168. Schemaname kann aber auch separat angegeben werden. Ein Schemaname, der mit dem
  169. Tabellennamen angegeben wurde, bekommt Vorrang, falls beides angegeben wurde.
  170. </para>
  171. <example id="zend.db.select.building.from.example-schema">
  172. <title>Beispiel für das Angeben eines Schemanamens</title>
  173. <programlisting language="php"><![CDATA[
  174. // Erzeut diese Anfrage:
  175. // SELECT *
  176. // FROM "myschema"."products"
  177. $select = $db->select()
  178. ->from('myschema.products');
  179. // oder
  180. $select = $db->select()
  181. ->from('products', '*', 'myschema');
  182. ]]></programlisting>
  183. </example>
  184. </sect3>
  185. <sect3 id="zend.db.select.building.columns">
  186. <title>Hinzufügen von Spalten</title>
  187. <para>
  188. Im zweiten Argument der Methode <methodname>from()</methodname> kann angegeben
  189. werden, welche Spalten der Tabelle ausgelesen werden sollen. Werden keine
  190. Spalten angegeben, so gilt der Standardwert <emphasis>*</emphasis>, der
  191. <acronym>SQL</acronym> Platzhalter für alle Spalten.
  192. </para>
  193. <para>
  194. Die Spalten können in einem einfachen Array von Strings oder einem assoziativen
  195. Array, in dem Aliasnamen den Spaltennamen zugewiesen werden, angegeben werden. Soll
  196. nur eine einzelne Spalte ohne Aliasnamen ausgelesen werden, so kann auch ein
  197. einfacher String übergeben werden.
  198. </para>
  199. <para>
  200. Wird ein leeres Array übergeben, so werden auch keine Spalten der Tabelle in den
  201. Ergebnissatz aufgenommen. Ein Codebeispiel gibt es unter
  202. <link linkend="zend.db.select.building.join.example-no-columns">Codebeispiel</link>
  203. bei der Methode <methodname>join()</methodname>.
  204. </para>
  205. <para>
  206. Der Spaltenname kann mit "<command>beziehungsName.spaltenName</command>" angegeben
  207. werden. <classname>Zend_Db_Select</classname> quotiert die einzelnen Teile für sich.
  208. Wird kein Beziehungsname für die Spalte angegeben, dann wird der Beziehungsname der
  209. Tabelle der aktuellen Methode <methodname>from()</methodname> verwendet.
  210. </para>
  211. <example id="zend.db.select.building.columns.example">
  212. <title>Beispiele für das Angeben von Spalten</title>
  213. <programlisting language="php"><![CDATA[
  214. // Erzeugt diese Anfrage:
  215. // SELECT p."product_id", p."product_name"
  216. // FROM "products" AS p
  217. $select = $db->select()
  218. ->from(array('p' => 'products'),
  219. array('product_id', 'product_name'));
  220. // Erzeugt dieselbe Anfrage, Angabe von Beziehungsnamen:
  221. // SELECT p."product_id", p."product_name"
  222. // FROM "products" AS p
  223. $select = $db->select()
  224. ->from(array('p' => 'products'),
  225. array('p.product_id', 'p.product_name'));
  226. // Erzeugt diese Anfrage mit einem Alias für eine Spalte:
  227. // SELECT p."product_id" AS prodno, p."product_name"
  228. // FROM "products" AS p
  229. $select = $db->select()
  230. ->from(array('p' => 'products'),
  231. array('prodno' => 'product_id', 'product_name'));
  232. ]]></programlisting>
  233. </example>
  234. </sect3>
  235. <sect3 id="zend.db.select.building.columns-expr">
  236. <title>Hinzufügen von Spalten mit Ausdrücke</title>
  237. <para>
  238. Spalten in einer <acronym>SQL</acronym>-Anfrage sind manchmal Ausdrücke, keine
  239. einfachen Spaltennamen einer Tabelle. Ausdrücke dürfen keine Beziehungsnamen oder
  240. Quotierungen bekommen. Wenn der Spaltenstring runde Klammern enthält, erkennt
  241. <classname>Zend_Db_Select</classname> dies als Ausdruck.
  242. </para>
  243. <para>
  244. Es kann auch ein Objekt des Typs <classname>Zend_Db_Expr</classname> erzeugt werden
  245. um zu verhindern, dass ein String wie ein Spaltenname behandelt wird.
  246. <classname>Zend_Db_Expr</classname> ist eine Minimalklasse, die einen String enthält.
  247. <classname>Zend_Db_Select</classname> erkennt Objekte des Typs
  248. <classname>Zend_Db_Expr</classname> und konvertiert diese in Strings, nimmt aber
  249. keine Änderungen daran vor, wie Quotierung oder Beziehungsnamen.
  250. </para>
  251. <note>
  252. <para>
  253. Die Benutzung von <classname>Zend_Db_Expr</classname> für Spaltennamen ist nicht
  254. nötig, wenn die Spaltennamen Ausdrücke runde Klammern enthalten.
  255. <classname>Zend_Db_Select</classname> erkennt diese und behandelt den String als
  256. Ausdruck und lässt Quotierung und Beziehungsnamen aus.
  257. </para>
  258. </note>
  259. <example id="zend.db.select.building.columns-expr.example">
  260. <title>Beispiel für das Angeben von Spaltennamen, die Ausdrücke enthalten</title>
  261. <programlisting language="php"><![CDATA[
  262. // Erzeugt diese Anfrage:
  263. // SELECT p."product_id", LOWER(product_name)
  264. // FROM "products" AS p
  265. // Ein Ausdruck eingeschlossen von runden Klammern wird zu Zend_Db_Expr.
  266. $select = $db->select()
  267. ->from(array('p' => 'products'),
  268. array('product_id', 'LOWER(product_name)'));
  269. // Erzeugt diese Anfrage:
  270. // SELECT p."product_id", (p.cost * 1.08) AS cost_plus_tax
  271. // FROM "products" AS p
  272. $select = $db->select()
  273. ->from(array('p' => 'products'),
  274. array('product_id',
  275. 'cost_plus_tax' => '(p.cost * 1.08)'));
  276. // Erzeugt diese Anfrage unter ausdrücklicher Verwendung von Zend_Db_Expr:
  277. // SELECT p."product_id", p.cost * 1.08 AS cost_plus_tax
  278. // FROM "products" AS p
  279. $select = $db->select()
  280. ->from(array('p' => 'products'),
  281. array('product_id',
  282. 'cost_plus_tax' =>
  283. new Zend_Db_Expr('p.cost * 1.08'))
  284. );
  285. ]]></programlisting>
  286. </example>
  287. <para>
  288. In den oben stehenden Fällen ändert <classname>Zend_Db_Select</classname> den String
  289. nicht mit Beziehungsnamen oder Bezeichnerquotierung. Wenn diese Änderungen notwendig
  290. sein sollten um Doppeldeutigkeiten aufzulösen, muss dies manuell am String
  291. geändert werden.
  292. </para>
  293. <para>
  294. Wenn die Spaltennamen aus <acronym>SQL</acronym>-Schlüsselwörtern bestehen oder
  295. spezielle Zeichen enthalten, sollte die Methode
  296. <methodname>quoteIdentifier()</methodname> verwendet werden und der Rückgabewert in
  297. den String eingefügt werden. Die Methode <methodname>quoteIdentifier()</methodname>
  298. verwendet <acronym>SQL</acronym>-Quotierung um Bezeichner abzugrenzen, wodurch klar
  299. wird, dass es sich um einen Bezeichner für eine Tabelle oder Spalte handelt, und
  300. nicht um einen anderen Teil der <acronym>SQL</acronym>-Syntax.
  301. </para>
  302. <para>
  303. Der Code wird datenbankunabhängiger, wenn die Methode
  304. <methodname>quoteIdentifier()</methodname> anstelle von direkter Eingabe
  305. der Quotierungszeichen verwendet wird, da einige <acronym>RDBMS</acronym>-Marken
  306. nicht-Standard Symbole für die Quotierung von Bezeichnern verwenden. Die
  307. Methode <methodname>quoteIdentifier()</methodname> wählt die passenden
  308. Quotierungssymbole für den Adaptertyp aus. Die
  309. Methode <methodname>quoteIdentifier()</methodname> ersetzt außerdem alle
  310. Quotierungszeichen innerhalb des Bezeichners.
  311. </para>
  312. <example id="zend.db.select.building.columns-quoteid.example">
  313. <title>Beispiel für die Quotierung von Spalten in einem Ausdruck</title>
  314. <programlisting language="php"><![CDATA[
  315. // Erzeugt folgende Anfrage und quotiert dabei einen Spaltennamen
  316. // "from" im Ausdruck:
  317. // SELECT p."from" + 10 AS origin
  318. // FROM "products" AS p
  319. $select = $db->select()
  320. ->from(array('p' => 'products'),
  321. array('origin' =>
  322. '(p.' . $db->quoteIdentifier('from') . ' + 10)')
  323. );
  324. ]]></programlisting>
  325. </example>
  326. </sect3>
  327. <sect3 id="zend.db.select.building.columns-atomic">
  328. <title>Spalten zu einer existierenden FROM oder JOIN Tabelle hinzufügen</title>
  329. <para>
  330. Es kann Fälle geben, in denen es gewünscht ist, Spalten zu einer bestehenden
  331. <acronym>FROM</acronym>- oder <acronym>JOIN</acronym>-Tabelle hinzuzufügen, nachdem
  332. diese Methoden aufgerufen wurden. Die Methode <methodname>columns()</methodname>
  333. erlaubt es spezifische Spalten an jedem Punkt hinzuzufügen, bevor die Abfrage
  334. aufgeführt wird. Die Spalte kann entweder als String oder
  335. <classname>Zend_Db_Expr</classname> oder als Array dieser Elemente angegeben werden.
  336. Das zweite Argument dieser Methode kann unterdrückt werden, was impliziert, dass die
  337. Spalten zu der <acronym>FROM</acronym>-Tabelle hinzugefügt werden sollen,
  338. andernfall muß ein bestehender Korrelationsname verwendet werden.
  339. </para>
  340. <example id="zend.db.select.building.columns-atomic.example">
  341. <title>Beispiel für das Hinzufügen von Spalten mit der Methode columns()</title>
  342. <programlisting language="php"><![CDATA[
  343. // Diese Abfrage bauen:
  344. // SELECT p."product_id", p."product_name"
  345. // FROM "products" AS p
  346. $select = $db->select()
  347. ->from(array('p' => 'products'), 'product_id')
  348. ->columns('product_name');
  349. // Dieselbe Abfrage bauen, durch Angabe der Korrelationsnamen:
  350. // SELECT p."product_id", p."product_name"
  351. // FROM "products" AS p
  352. $select = $db->select()
  353. ->from(array('p' => 'products'), 'p.product_id')
  354. ->columns('product_name', 'p');
  355. // Alternativ kann columns('p.product_name') verwendet werden
  356. ]]></programlisting>
  357. </example>
  358. </sect3>
  359. <sect3 id="zend.db.select.building.join">
  360. <title>Hinzufügen einer weiteren Tabelle zu der Anfrage mit JOIN</title>
  361. <para>
  362. Viele nützliche Anfragen benötigen ein <acronym>JOIN</acronym> um mehrere Spalten
  363. verschiedener Tabellen miteinander zu kombinieren. Tabellen können zu einer
  364. <classname>Zend_Db_Select</classname>-Anfrage mit der
  365. Methode <methodname>join()</methodname> hinzugefügt werden. Die Nutzung dieser
  366. Methode ist ähnlich der Methode <methodname>from()</methodname>, außer dass in den
  367. meisten Fällen zusätzlich eine Join-Bedingung angegeben werden kann.
  368. </para>
  369. <example id="zend.db.select.building.join.example">
  370. <title>Beispiel für die Methode join()</title>
  371. <programlisting language="php"><![CDATA[
  372. // Erzeugt diese Anfrage:
  373. // SELECT p."product_id", p."product_name", l.*
  374. // FROM "products" AS p JOIN "line_items" AS l
  375. // ON p.product_id = l.product_id
  376. $select = $db->select()
  377. ->from(array('p' => 'products'),
  378. array('product_id', 'product_name'))
  379. ->join(array('l' => 'line_items'),
  380. 'p.product_id = l.product_id');
  381. ]]></programlisting>
  382. </example>
  383. <para>
  384. Das zweite Argument der Methode <methodname>join()</methodname> ist ein String mit
  385. der Join-Bedingung. Dies ist ein Ausdruck die Kriterien angibt, welche Zeilen
  386. in der einen Tabelle mit Zeilen einer anderen Tabelle verknüpft werden. Es können
  387. Beziehungsnamen in diesem Ausdruck verwendet werden.
  388. </para>
  389. <note>
  390. <para>
  391. Es wird keine Quotierung auf den Ausdruck für die Join-Bedingung angewendet.
  392. Werden Spaltennamen verwendet, welche Quotierung benötigen, so muss
  393. <methodname>quoteIdentifier()</methodname> verwendet werden, wenn der String für
  394. die Join-Bedingung formuliert wird.
  395. </para>
  396. </note>
  397. <para>
  398. Das dritte Argument für <methodname>join()</methodname> ist ein Array von
  399. Spaltennamen, entsprechend des Arrays der Methode <methodname>from()</methodname>.
  400. Der Standard ist ebenfalls "<emphasis>*</emphasis>" und unterstützt Beziehungsnamen,
  401. Ausdrücke und <classname>Zend_Db_Expr</classname> in der gleichen Weise wie dem
  402. Array von Spaltennamen der Methode <methodname>from()</methodname>.
  403. </para>
  404. <para>
  405. Wenn keine Spalten einer Tabelle ausgewählt werden soll, muss ein leeres Array für
  406. die Liste der Spaltennamen übergeben werden. Diese Nutzung funktioniert ebenfalls
  407. in der Methode <methodname>from()</methodname>, aber normalerweise werden einige
  408. Spalten der primären Tabelle in den Anfragen benötigt, während möglicherweise keine
  409. Spalten der verbundenen Tabelle ausgewählt werden sollen.
  410. </para>
  411. <example id="zend.db.select.building.join.example-no-columns">
  412. <title>Beispiel für das Angeben keiner Spalten</title>
  413. <programlisting language="php"><![CDATA[
  414. // Erzeugt diese Anfrage:
  415. // SELECT p."product_id", p."product_name"
  416. // FROM "products" AS p JOIN "line_items" AS l
  417. // ON p.product_id = l.product_id
  418. $select = $db->select()
  419. ->from(array('p' => 'products'),
  420. array('product_id', 'product_name'))
  421. ->join(array('l' => 'line_items'),
  422. 'p.product_id = l.product_id',
  423. array() ); // leere Liste von Spalten
  424. ]]></programlisting>
  425. <para>
  426. Zu beachten ist, dass leere <methodname>array()</methodname> in dem oben
  427. stehenden Beispiel, an der Stelle einer Liste von Spalten der verbundenen
  428. Tabelle.
  429. </para>
  430. </example>
  431. <para>
  432. <acronym>SQL</acronym> kennt verschiedene Varianten von Joins. In der Liste weiter
  433. unten sind Methoden für die verschiedenen Join-Varianten zu finden, die
  434. <classname>Zend_Db_Select</classname> unterstützt.
  435. </para>
  436. <itemizedlist>
  437. <listitem>
  438. <para>
  439. <command>INNER JOIN</command> mit den Methoden
  440. <methodname>join(tabelle, join, [spalten])</methodname> oder
  441. <methodname>joinInner(tabelle, join, [spalten])</methodname>.
  442. </para>
  443. <para>
  444. Dies wird der gebräuchlichste Typ von Join sein. Zeilen jeder Tabelle
  445. werden mit Hilfe der angegebenen Join-Bedingung verglichen. Der
  446. Ergebnissatz enthält nur die Zeilen, die der Join Bedingungen entsprechen.
  447. Der Ergebnissatz kann leer sein, wenn keine Zeile die Bedingung erfüllt.
  448. </para>
  449. <para>
  450. Alle <acronym>RDBMS</acronym>-Marken unterstützen diese Variante.
  451. </para>
  452. </listitem>
  453. <listitem>
  454. <para>
  455. <command>LEFT JOIN</command> mit der Methode
  456. <methodname>joinLeft(tabelle, bedingung, [spalten])</methodname>.
  457. </para>
  458. <para>
  459. Alle Zeilen der links vom Operanden stehenden Tabelle sind enthalten,
  460. passende Zeilen der rechts stehenden Tabelle sind ebenfalls enthalten. Die
  461. Spalten der rechts stehenden Tabelle werden mit <constant>NULL</constant>
  462. aufgefüllt, wenn keine zu der linken Tabelle passenden Zeilen existieren.
  463. </para>
  464. <para>
  465. Alle <acronym>RDBMS</acronym>-Marken unterstützen diese Variante.
  466. </para>
  467. </listitem>
  468. <listitem>
  469. <para>
  470. <command>RIGHT JOIN</command> mit der Methode
  471. <methodname>joinRight(tabelle, bedingung, [spalten])</methodname>.
  472. </para>
  473. <para>
  474. Right Outer Join ist das Gegenstück zu Left Outer Join. Alle Zeilen der
  475. rechts vom Operanden stehenden Tabelle sind enthalten, passende Zeilen der
  476. links stehenden Tabelle sind ebenfalls enthalten. Die Spalten der links
  477. stehenden Tabelle werden mit <constant>NULL</constant> aufgefüllt, wenn
  478. keine zu der rechten Tabelle passenden Zeilen existieren.
  479. </para>
  480. <para>
  481. Einige <acronym>RDBMS</acronym>-Marken unterstützen diesen Join nicht,
  482. aber grundsätzlich kann jeder Right Join durch einen Left Join mit
  483. umgekehrter Sortierung der Tabellen dargestellt werden.
  484. </para>
  485. </listitem>
  486. <listitem>
  487. <para>
  488. <command>FULL JOIN</command> mit der Methode
  489. <methodname>joinFull(tabelle, bedingung, [spalten])</methodname>.
  490. </para>
  491. <para>
  492. Ein Full Outer Join ist wie eine Kombination eines Left Outer Join mit
  493. einem Right Outer Join. Alle Zeilen beider Tabellen sind enthalten, gepaart
  494. miteinander in der gleichen Zeile des Ergebnissatzes wenn die
  495. Join-Bedingung erfüllt wird, oder wenn nicht, mit <constant>NULL</constant>
  496. an Stelle der Spalten der anderen Tabelle.
  497. </para>
  498. <para>
  499. Einige <acronym>RDBMS</acronym>-Marken unterstützen diesen Join nicht.
  500. </para>
  501. </listitem>
  502. <listitem>
  503. <para>
  504. <command>CROSS JOIN</command> mit der Methode
  505. <methodname>joinCross(tabelle, [spalten])</methodname>.
  506. </para>
  507. <para>
  508. Ein Cross Join ist ein Kartesisches Produkt. Jede Zeile der ersten Tabelle
  509. wird mit jeder Zeile der zweiten Tabelle verbunden. Daher ist die Anzahl
  510. der Zeilen im Ergebnissatz gleich dem Produkt der Zeilenanzahlen der beiden
  511. Tabellen. Der Ergebnissatz kann mit Bedingungen einer
  512. <acronym>WHERE</acronym>-Bedingung gefiltert werden. Ein Cross Join ist
  513. ähnlich der alten <acronym>SQL</acronym>-89 Join Syntax.
  514. </para>
  515. <para>
  516. Die Methode <methodname>joinCross()</methodname> hat keinen Parameter für
  517. die Join-Bedingung. Einige <acronym>RDBMS</acronym>-Marken unterstützen
  518. diesen Join nicht.
  519. </para>
  520. </listitem>
  521. <listitem>
  522. <para>
  523. <command>NATURAL JOIN</command> mit der Methode
  524. <methodname>joinNatural(tabelle, [spalten])</methodname>.
  525. </para>
  526. <para>
  527. Ein Natural Join vergleicht alle Spalten die in beiden Tabellen mit
  528. gleichem Namen vorkommen. Der Vergleich prüft Gleichheit aller Spalten, ein
  529. Vergleich auf Ungleichheit ist kein Natural Join. Von dieser
  530. <acronym>API</acronym> werden nur Natural Inner Joins unterstützt, auch wenn
  531. <acronym>SQL</acronym> auch Natural Outer Joins erlaubt.
  532. </para>
  533. <para>
  534. Die Methode <methodname>joinNatural()</methodname> hat keinen Parameter für
  535. die Join-Bedingung.
  536. </para>
  537. </listitem>
  538. </itemizedlist>
  539. <para>
  540. Zusätzlich zu diesen Join-Methoden können Abfragen durch Verwendung der Methoden
  541. JoinUsing vereinfacht werden. Statt dass eine komplette Definition des Joins
  542. angegeben wird, kann einfach der Spaltenname übergeben werden, auf welchem gejoint
  543. werden soll und das <classname>Zend_Db_Select</classname>-Objekt vervollständigt die
  544. Bedingung alleine.
  545. </para>
  546. <example id="zend.db.select.building.joinusing.example">
  547. <title>Beispiel für die Methode joinUsing()</title>
  548. <programlisting language="php"><![CDATA[
  549. // Erzeugt diese Abfrage
  550. // SELECT *
  551. // FROM "table1"
  552. // JOIN "table2"
  553. // ON "table1".column1 = "table2".column1
  554. // WHERE column2 = 'foo'
  555. $select = $db->select()
  556. ->from('table1')
  557. ->joinUsing('table2', 'column1')
  558. ->where('column2 = ?', 'foo');
  559. ]]></programlisting>
  560. </example>
  561. <para>
  562. Jede der anwendbaren Join-Methoden in der Komponente
  563. <classname>Zend_Db_Select</classname> hat eine entsprechende 'using'-Methode.
  564. </para>
  565. <itemizedlist>
  566. <listitem>
  567. <para>
  568. <methodname>joinUsing(table, join, [columns])</methodname> und
  569. <methodname>joinInnerUsing(table, join, [columns])</methodname>
  570. </para>
  571. </listitem>
  572. <listitem>
  573. <para>
  574. <methodname>joinLeftUsing(table, join, [columns])</methodname>
  575. </para>
  576. </listitem>
  577. <listitem>
  578. <para>
  579. <methodname>joinRightUsing(table, join, [columns])</methodname>
  580. </para>
  581. </listitem>
  582. <listitem>
  583. <para>
  584. <methodname>joinFullUsing(table, join, [columns])</methodname>
  585. </para>
  586. </listitem>
  587. </itemizedlist>
  588. </sect3>
  589. <sect3 id="zend.db.select.building.where">
  590. <title>Hinzufügen eines WHERE-Abschnitts</title>
  591. <para>
  592. Mit der Methode <methodname>where()</methodname> können Kriterien angegeben werden,
  593. die den Ergebnissatz einschränken. Das erste Argument
  594. dieser Methode ist ein <acronym>SQL</acronym>-Ausdruck, welche in einer
  595. <acronym>SQL</acronym>-<acronym>WHERE</acronym>-Klausel der Anfrage steht.
  596. </para>
  597. <example id="zend.db.select.building.where.example">
  598. <title>Beispiel für die Methode where()</title>
  599. <programlisting language="php"><![CDATA[
  600. // Erzeugt diese Anfrage:
  601. // SELECT product_id, product_name, price
  602. // FROM "products"
  603. // WHERE price > 100.00
  604. $select = $db->select()
  605. ->from('products',
  606. array('product_id', 'product_name', 'price'))
  607. ->where('price > 100.00');
  608. ]]></programlisting>
  609. </example>
  610. <note>
  611. <para>
  612. Auf Ausdrücke, welche an die Methoden <methodname>where()</methodname> oder
  613. <methodname>orWhere()</methodname> übergeben werden, wird keine
  614. Quotierung angewendet. Werden Spaltennamen verwendet, die quotiert werden
  615. müssen, so muss <methodname>quoteIdentifier()</methodname> verwendet werden,
  616. wenn der String für die Bedingung formuliert wird.
  617. </para>
  618. </note>
  619. <para>
  620. Das zweite Argument der Methode <methodname>where()</methodname> ist optional. Es
  621. ist ein Wert, der in den Ausdruck eingesetzt wird.
  622. <classname>Zend_Db_Select</classname> quotiert den Wert und ersetzt ihn für ein
  623. Fragezeichen ("<emphasis>?</emphasis>") im Ausdruck.
  624. </para>
  625. <example id="zend.db.select.building.where.example-param">
  626. <title>Beispiel für einen Parameter in der Methode where()</title>
  627. <programlisting language="php"><![CDATA[
  628. // Erzeugt diese Anfrage:
  629. // SELECT product_id, product_name, price
  630. // FROM "products"
  631. // WHERE (price > 100.00)
  632. $minimumPrice = 100;
  633. $select = $db->select()
  634. ->from('products',
  635. array('product_id', 'product_name', 'price'))
  636. ->where('price > ?', $minimumPrice);
  637. ]]></programlisting>
  638. </example>
  639. <para>
  640. Man kann als zweiten Parameter ein Array an die Methode
  641. <methodname>where()</methodname> übergeben, wenn der <acronym>SQL</acronym>-Operator
  642. IN verwendet wird.
  643. </para>
  644. <example id="zend.db.select.building.where.example-array">
  645. <title>Beispiel mit Array als Parameters in der Methode where()</title>
  646. <programlisting language="php"><![CDATA[
  647. // Diese Abrage wird gebaut:
  648. // SELECT product_id, product_name, price
  649. // FROM "products"
  650. // WHERE (product_id IN (1, 2, 3))
  651. $productIds = array(1, 2, 3);
  652. $select = $db->select()
  653. ->from('products',
  654. array('product_id', 'product_name', 'price'))
  655. ->where('product_id IN (?)', $productIds);
  656. ]]></programlisting>
  657. </example>
  658. <para>
  659. Die Methode <methodname>where()</methodname> kann mehrere Male in demselben
  660. <classname>Zend_Db_Select</classname>-Objekt aufgerufen werden. Die daraus folgende
  661. Anfrage kombiniert die unterschiedlichen Ausdrücke unter Benutzung von
  662. <acronym>AND</acronym> dazwischen.
  663. </para>
  664. <example id="zend.db.select.building.where.example-and">
  665. <title>Beispiel für Mehrfachaufruf der Methode where()</title>
  666. <programlisting language="php"><![CDATA[
  667. // Erzeugt diese Anfrage:
  668. // SELECT product_id, product_name, price
  669. // FROM "products"
  670. // WHERE (price > 100.00)
  671. // AND (price < 500.00)
  672. $minimumPrice = 100;
  673. $maximumPrice = 500;
  674. $select = $db->select()
  675. ->from('products',
  676. array('product_id', 'product_name', 'price'))
  677. ->where('price > ?', $minimumPrice)
  678. ->where('price < ?', $maximumPrice);
  679. ]]></programlisting>
  680. </example>
  681. <para>
  682. Wenn mehrere Ausdrücke mit <acronym>OR</acronym> verknüpft werden sollen, kann die
  683. Methode <methodname>orWhere()</methodname> verwendet werden. Sie wird genauso
  684. benutzt wie die Methode <methodname>where()</methodname>, außer dass dem angegebenen
  685. Ausdruck ein <acronym>OR</acronym> vorangestellt wird, anstelle eines
  686. <acronym>AND</acronym>.
  687. </para>
  688. <example id="zend.db.select.building.where.example-or">
  689. <title>Beispiel für die Methode orWhere()</title>
  690. <programlisting language="php"><![CDATA[
  691. // Erzeugt diese Anfrage:
  692. // SELECT product_id, product_name, price
  693. // FROM "products"
  694. // WHERE (price < 100.00)
  695. // OR (price > 500.00)
  696. $minimumPrice = 100;
  697. $maximumPrice = 500;
  698. $select = $db->select()
  699. ->from('products',
  700. array('product_id', 'product_name', 'price'))
  701. ->where('price < ?', $minimumPrice)
  702. ->orWhere('price > ?', $maximumPrice);
  703. ]]></programlisting>
  704. </example>
  705. <para>
  706. <classname>Zend_Db_Select</classname> klammert Ausdrücke automatisch mit runden
  707. Klammern ein, wenn sie mit einer der Methoden <methodname>where()</methodname> oder
  708. <methodname>orWhere()</methodname> erzeugt wurden. Dies hilft sicherzustellen,
  709. dass das Voranstellen von Boolschen Operatoren keine unerwarteten Ergebnisse
  710. nach sich zieht.
  711. </para>
  712. <example id="zend.db.select.building.where.example-parens">
  713. <title>Beispiel für das Einklammern von Boolschen Ausdrücken</title>
  714. <programlisting language="php"><![CDATA[
  715. // Erzeugt diese Anfrage:
  716. // SELECT product_id, product_name, price
  717. // FROM "products"
  718. // WHERE (price < 100.00 OR price > 500.00)
  719. // AND (product_name = 'Apple')
  720. $minimumPrice = 100;
  721. $maximumPrice = 500;
  722. $prod = 'Apple';
  723. $select = $db->select()
  724. ->from('products',
  725. array('product_id', 'product_name', 'price'))
  726. ->where("price < $minimumPrice OR price > $maximumPrice")
  727. ->where('product_name = ?', $prod);
  728. ]]></programlisting>
  729. </example>
  730. <para>
  731. Im oben stehenden Beispiel wären die Ergebnisse ohne die Klammern ziemlich
  732. anders, weil <acronym>AND</acronym> eine höhere Priorität als <acronym>OR</acronym>
  733. hat. <classname>Zend_Db_Select</classname> erzeugt runde Klammern, wodurch jeder
  734. Ausdruck von aufeinander folgenden Aufrufen der Methode
  735. <methodname>where()</methodname> fester binden als das <acronym>AND</acronym>,
  736. welches die Ausdrücke kombiniert.
  737. </para>
  738. </sect3>
  739. <sect3 id="zend.db.select.building.group">
  740. <title>Hinzufügen eines GROUP BY-Abschnitts</title>
  741. <para>
  742. In <acronym>SQL</acronym> ermöglicht der <command>GROUP BY</command>-Abschnitt die
  743. Zeilenzahl des Ergebnissatzes auf eine Zeile pro eindeutigem Wert der Spalte(n)
  744. einzuschränken, welche im <command>GROUP BY</command>-Abschnitt benannt sind.
  745. </para>
  746. <para>
  747. In <classname>Zend_Db_Select</classname> können diese Spalte(n) mit der
  748. Methode <methodname>group()</methodname> angegeben werden. Das Argument der Methode
  749. ist ein Spaltenname oder ein Array von Spaltennamen, welche im
  750. <command>GROUP BY</command>-Abschnitt stehen sollen.
  751. </para>
  752. <example id="zend.db.select.building.group.example">
  753. <title>Beispiel für die Methode group()</title>
  754. <programlisting language="php"><![CDATA[
  755. // Erzeugt diese Anfrage:
  756. // SELECT p."product_id", COUNT(*) AS line_items_per_product
  757. // FROM "products" AS p JOIN "line_items" AS l
  758. // ON p.product_id = l.product_id
  759. // GROUP BY p.product_id
  760. $select = $db->select()
  761. ->from(array('p' => 'products'),
  762. array('product_id'))
  763. ->join(array('l' => 'line_items'),
  764. 'p.product_id = l.product_id',
  765. array('line_items_per_product' => 'COUNT(*)'))
  766. ->group('p.product_id');
  767. ]]></programlisting>
  768. </example>
  769. <para>
  770. Wie in dem Array von Spaltennamen der Methode <methodname>from()</methodname>, so
  771. können auch hier Beziehungsnamen in den Strings der Spaltennamen verwendet werden,
  772. und der Spaltenname wird als Bezeichner quotiert, wenn er nicht in runden Klammern
  773. steht oder ein Objekt des Typs <classname>Zend_Db_Expr</classname> ist.
  774. </para>
  775. </sect3>
  776. <sect3 id="zend.db.select.building.having">
  777. <title>Hinzufügen eines HAVING-Abschnittes</title>
  778. <para>
  779. In <acronym>SQL</acronym> fügt der <constant>HAVING</constant>-Abschnitt eine
  780. Beschränkungsbedingung für Gruppen von Zeilen ein. Dies ist ähnlich der
  781. Einschränkungsbedingungen auf Zeilen, des <constant>WHERE</constant>-Abschnittes.
  782. Die beiden Abschnitte unterscheiden sich jedoch, denn die <constant>WHERE</constant>
  783. Bedingungen werden angewendet, bevor Gruppen definiert wurden. Im Gegensatz werden
  784. <constant>HAVING</constant>-Bedingungen erst angewendet, nachdem Gruppen definiert
  785. wurden.
  786. </para>
  787. <para>
  788. In <classname>Zend_Db_Select</classname> können Bedingungen für die Einschränkung
  789. von Gruppen mit der Methode <methodname>having()</methodname> angegeben werden. Die
  790. Nutzung ist ähnlich wie die der Methode <methodname>where()</methodname>. Das erste
  791. Argument ist ein String, welcher einen <acronym>SQL</acronym>-Ausdruck enthält. Das
  792. zweite Argument ist optional und wird verwendet, um einen
  793. Platzhalter im <acronym>SQL</acronym>-Ausdruck zu ersetzen. Ausdrücke, die
  794. durch mehrfaches Aufrufen der Methode <methodname>having()</methodname> erzeugt
  795. wurden, werden mit dem Boolschen Operator <acronym>AND</acronym> verknüpft, oder mit
  796. dem Operator <acronym>OR</acronym>, wenn die Methode
  797. <methodname>orHaving()</methodname> verwendet wird.
  798. </para>
  799. <example id="zend.db.select.building.having.example">
  800. <title>Beispiel für die Methode having()</title>
  801. <programlisting language="php"><![CDATA[
  802. // Erzeugt diese Anfrage:
  803. // SELECT p."product_id", COUNT(*) AS line_items_per_product
  804. // FROM "products" AS p JOIN "line_items" AS l
  805. // ON p.product_id = l.product_id
  806. // GROUP BY p.product_id
  807. // HAVING line_items_per_product > 10
  808. $select = $db->select()
  809. ->from(array('p' => 'products'),
  810. array('product_id'))
  811. ->join(array('l' => 'line_items'),
  812. 'p.product_id = l.product_id',
  813. array('line_items_per_product' => 'COUNT(*)'))
  814. ->group('p.product_id')
  815. ->having('line_items_per_product > 10');
  816. ]]></programlisting>
  817. </example>
  818. <note>
  819. <para>
  820. Es wird keine Quotierung auf Ausdrücke angewendet, welche an die Methoden
  821. <methodname>having()</methodname> oder <methodname>orHaving()</methodname>
  822. übergeben werden. Werden Spaltennamen verwendet, die quotiert werden
  823. müssen, so muss <methodname>quoteIdentifier()</methodname> verwendet werden,
  824. wenn der String für die Bedingung formuliert wird.
  825. </para>
  826. </note>
  827. </sect3>
  828. <sect3 id="zend.db.select.building.order">
  829. <title>Hinzufügen eines ORDER BY Abschnitts</title>
  830. <para>
  831. In <acronym>SQL</acronym> gibt der <acronym>ORDER</acronym> BY Abschnitt eine oder
  832. mehrere Spalten oder Ausdrücke an, wonach ein Ergebnissatz sortiert wird. Wenn
  833. mehrere Spalten angegeben sind, werden die sekundären Spalten verwendet um "ties"
  834. aufzulösen; die Sortierung wird von sekundären Spalten bestimmt, wenn vorhergehende
  835. Spalten identische Werte enthalten. Die Standardsortierung ist vom kleinsten zum
  836. größten Wert. Dieses Verhalten kann umgekehrt werden, wenn das Schlüsselwort
  837. <constant>DESC</constant> nach der Spalte angegeben wird.
  838. </para>
  839. <para>
  840. In <classname>Zend_Db_Select</classname> kann die Methode
  841. <methodname>order()</methodname> verwendet werden um Spalten oder Arrays von Spalten
  842. anzugeben, nach denen sortiert werden soll. Jedes Element des Arrays ist ein String,
  843. welcher die Spalte benennt. Optional kann auf den Namen eines der Schlüsselwörter
  844. <constant>ASC</constant> <constant>DESC</constant> getrennt durch ein
  845. Leerzeichen folgen.
  846. </para>
  847. <para>
  848. Wie in den Methoden <methodname>from()</methodname> und
  849. <methodname>group()</methodname> werden Spalten als Bezeichner quotiert, wenn sie
  850. nicht von runden Klammern eingeschlossen oder vom Objekttyp
  851. <classname>Zend_Db_Expr</classname> sind.
  852. </para>
  853. <example id="zend.db.select.building.order.example">
  854. <title>Beispiel für die Methode order()</title>
  855. <programlisting language="php"><![CDATA[
  856. // Erzeugt diese Anfrage:
  857. // SELECT p."product_id", COUNT(*) AS line_items_per_product
  858. // FROM "products" AS p JOIN "line_items" AS l
  859. // ON p.product_id = l.product_id
  860. // GROUP BY p.product_id
  861. // ORDER BY "line_items_per_product" DESC, "product_id"
  862. $select = $db->select()
  863. ->from(array('p' => 'products'),
  864. array('product_id'))
  865. ->join(array('l' => 'line_items'),
  866. 'p.product_id = l.product_id',
  867. array('line_items_per_product' => 'COUNT(*)'))
  868. ->group('p.product_id')
  869. ->order(array('line_items_per_product DESC',
  870. 'product_id'));
  871. ]]></programlisting>
  872. </example>
  873. </sect3>
  874. <sect3 id="zend.db.select.building.limit">
  875. <title>Hinzufügen eines LIMIT-Abschnitts</title>
  876. <para>
  877. Einige <acronym>RDBMS</acronym>-Marken erweitern <acronym>SQL</acronym> mit einem
  878. Anfrageabschnitt, bekannt als <constant>LIMIT</constant>-Abschnitt. Dieser
  879. Abschnitt begrenzt die Anzahl der Zeilen in einem Ergebnissatz auf die angegebene
  880. Höchstanzahl. Es kann ebenfalls angegeben werden, dass eine Anzahl von Zeilen
  881. ausgelassen werden soll. Dieses Feature erlaubt es, eine Untermenge des
  882. Ergebnissatzes zu holen, zum Beispiel wenn Anfrageergebnisse auf einander
  883. folgenden Seiten angezeigt werden sollen.
  884. </para>
  885. <para>
  886. In <classname>Zend_Db_Select</classname> kann die Methode
  887. <methodname>limit()</methodname> verwendet werden, um die Anzahl von Zeilen und die
  888. Anzahl der auszulassenden Spalten anzugeben. Das <emphasis>erste</emphasis> Argument
  889. dieser Methode ist die gewünschte Anzahl an Zeilen. Das <emphasis>zweite</emphasis>
  890. Argument gibt die Anzahl der auszulassenden Zeilen an.
  891. </para>
  892. <example id="zend.db.select.building.limit.example">
  893. <title>Beispiel für die Methode limit()</title>
  894. <programlisting language="php"><![CDATA[
  895. // Erzeugt diese Anfrage:
  896. // SELECT p."product_id", p."product_name"
  897. // FROM "products" AS p
  898. // LIMIT 10, 20
  899. // Identisch zu:
  900. // SELECT p."product_id", p."product_name"
  901. // FROM "products" AS p
  902. // LIMIT 20 OFFSET 10
  903. $select = $db->select()
  904. ->from(array('p' => 'products'),
  905. array('product_id', 'product_name'))
  906. ->limit(20, 10);
  907. ]]></programlisting>
  908. </example>
  909. <note>
  910. <para>
  911. Die <constant>LIMIT</constant>-Syntax wird nicht von allen
  912. <acronym>RDBMS</acronym>-Marken unterstützt. Einige <acronym>RDBMS</acronym>
  913. benötigen eine unterschiedliche Syntax für eine ähnliche Funktionalität. Jede
  914. <classname>Zend_Db_Adapter_Abstract</classname>-Klasse enthält eine Methode um
  915. die für das <acronym>RDBMS</acronym> passende <acronym>SQL</acronym>-Syntax zu
  916. erzeugen.
  917. </para>
  918. </note>
  919. <para>
  920. Die Methode <methodname>limitPage()</methodname> kann als alternativer Weg
  921. verwendet werden, um Zeilenanzahl und Offset zu anzugeben. Diese Methode erlaubt es,
  922. den Ergebnissatz auf ein Subset aus einer Serie von Subsets mit Reihen einer
  923. fixen Länge aus dem Gesamtergebnis zu begrenzen. In anderen
  924. Worten gibt man die Länge einer Seite ("page") und die
  925. Nummer der Seite an, die als Ergebnis von der Abfrage
  926. zurückgegeben werden sollen. Die Seitennummer ist das erste Argument der
  927. Methode <methodname>limitPage()</methodname>, die Seitenlänge ist das zweite
  928. Argument. Beide Argumente werden benötigt; sie haben keinen Standardwert.
  929. </para>
  930. <example id="zend.db.select.building.limit.example2">
  931. <title>Beispiel der Methode limitPage()</title>
  932. <programlisting language="php"><![CDATA[
  933. // Erstelle diese Abfrage:
  934. // SELECT p."product_id", p."product_name"
  935. // FROM "products" AS p
  936. // LIMIT 10, 20
  937. $select = $db->select()
  938. ->from(array('p' => 'products'),
  939. array('product_id', 'product_name'))
  940. ->limitPage(2, 10);
  941. ]]></programlisting>
  942. </example>
  943. </sect3>
  944. <sect3 id="zend.db.select.building.distinct">
  945. <title>Hinzufügen des Anfragewandlers DISTINCT</title>
  946. <para>
  947. Die Methode <methodname>distinct()</methodname> ermöglicht es, das
  948. Schlüsselwort <constant>DISTINCT</constant> in die <acronym>SQL</acronym>-Syntax
  949. einzufügen.
  950. </para>
  951. <example id="zend.db.select.building.distinct.example">
  952. <title>Beispiel für die Methode distinct()</title>
  953. <programlisting language="php"><![CDATA[
  954. // Erzeugt diese Anfrage:
  955. // SELECT DISTINCT p."product_name"
  956. // FROM "products" AS p
  957. $select = $db->select()
  958. ->distinct()
  959. ->from(array('p' => 'products'), 'product_name');
  960. ]]></programlisting>
  961. </example>
  962. </sect3>
  963. <sect3 id="zend.db.select.building.for-update">
  964. <title>Hinzufügen des Anfragewandlers FOR UPDATE</title>
  965. <para>
  966. Die Methode <methodname>forUpdate()</methodname> ermöglicht es, die
  967. Schlüsselworte <acronym>FOR</acronym> <acronym>UPDATE</acronym> in die
  968. <acronym>SQL</acronym>-Syntax einzufügen.
  969. </para>
  970. <example id="zend.db.select.building.for-update.example">
  971. <title>Beispiel der Methode forUpdate()</title>
  972. <programlisting language="php"><![CDATA[
  973. // Erzeugt diese Anfrage:
  974. // SELECT FOR UPDATE p.*
  975. // FROM "products" AS p
  976. $select = $db->select()
  977. ->forUpdate()
  978. ->from(array('p' => 'products'));
  979. ]]></programlisting>
  980. </example>
  981. </sect3>
  982. <sect3 id="zend.db.select.building.union">
  983. <title>Eine UNION-Abfrage erstellen</title>
  984. <para>
  985. Man kann Union-Abfragen mit <classname>Zend_Db_Select</classname> erstellen, indem
  986. ein Array von <classname>Zend_Db_Select</classname> oder <acronym>SQL</acronym>
  987. Query Strings an die Methode <methodname>union()</methodname> übergeben wird. Als
  988. zweiter Parameter können die Konstanten
  989. <constant>Zend_Db_Select::SQL_UNION</constant> oder
  990. <constant>Zend_Db_Select::SQL_UNION_ALL</constant> übergeben werden, um den Typ der
  991. Union zu anzugeben, den man ausführen will.
  992. </para>
  993. <example id="zend.db.select.building.union.example">
  994. <title>Beispiel der Methode union()</title>
  995. <programlisting language="php"><![CDATA[
  996. $sql1 = $db->select();
  997. $sql2 = "SELECT ...";
  998. $select = $db->select()
  999. ->union(array($sql1, $sql2))
  1000. ->order("id");
  1001. ]]></programlisting>
  1002. </example>
  1003. </sect3>
  1004. </sect2>
  1005. <sect2 id="zend.db.select.execute">
  1006. <title>Ausführen von SELECT-Anfragen</title>
  1007. <para>
  1008. Dieser Abschnitt beschreibt, wie Anfragen ausgeführt werden, die durch ein
  1009. <classname>Zend_Db_Select</classname>-Objekt repräsentiert werden.
  1010. </para>
  1011. <sect3 id="zend.db.select.execute.query-adapter">
  1012. <title>Ausführen von Select-Anfragen aus dem Db-Adapter</title>
  1013. <para>
  1014. Die Anfrage, die durch das <classname>Zend_Db_Select</classname>-Objekt
  1015. repräsentiert wird, kann ausgeführt werden, indem sie als erstes Argument an die
  1016. Methode <methodname>query()</methodname> des
  1017. <classname>Zend_Db_Adapter_Abstract</classname>-Objekts übergeben wird. Dabei wird
  1018. das <classname>Zend_Db_Select</classname> anstelle eines Strings verwendet.
  1019. </para>
  1020. <para>
  1021. Die Methode <methodname>query()</methodname> gibt ein Objekt vom Typ
  1022. <classname>Zend_Db_Statement</classname> oder PDOStatement
  1023. zurück, je nachdem welcher Adaptertyp verwendet wird.
  1024. </para>
  1025. <example id="zend.db.select.execute.query-adapter.example">
  1026. <title>Beispiel für die Nutzung der Methode query() des Db-Adapters</title>
  1027. <programlisting language="php"><![CDATA[
  1028. $select = $db->select()
  1029. ->from('products');
  1030. $stmt = $db->query($select);
  1031. $result = $stmt->fetchAll();
  1032. ]]></programlisting>
  1033. </example>
  1034. </sect3>
  1035. <sect3 id="zend.db.select.execute.query-select">
  1036. <title>Ausführen von Select-Anfragen mit dem Objekt</title>
  1037. <para>
  1038. Als Alternative zur Nutzung der Methode <methodname>query()</methodname> des
  1039. Adapterobjekts kann auch die Methode <methodname>query()</methodname> des
  1040. Objekts <classname>Zend_Db_Select</classname> verwendet werden. Beide Methoden
  1041. geben ein Objekt vom Typ <classname>Zend_Db_Statement</classname> oder
  1042. PDOStatement zurück, je nachdem welcher Adaptertyp verwendet wird.
  1043. </para>
  1044. <example id="zend.db.select.execute.query-select.example">
  1045. <title>Beispiel für die Nutzung der Methode query() des Select-Objekts</title>
  1046. <programlisting language="php"><![CDATA[
  1047. $select = $db->select()
  1048. ->from('products');
  1049. $stmt = $select->query();
  1050. $result = $stmt->fetchAll();
  1051. ]]></programlisting>
  1052. </example>
  1053. </sect3>
  1054. <sect3 id="zend.db.select.execute.tostring">
  1055. <title>Konvertieren eines Select-Objekts in einen SQL-String</title>
  1056. <para>
  1057. Wenn Zugriff auf die String-Repräsentation der <acronym>SQL</acronym>-Anfrage
  1058. benötigt wird, welche dem Objekt <classname>Zend_Db_Select</classname> entspricht,
  1059. kann die Methode <methodname>__toString()</methodname> verwendet werden.
  1060. </para>
  1061. <example id="zend.db.select.execute.tostring.example">
  1062. <title>Beispiel für die Methode __toString()</title>
  1063. <programlisting language="php"><![CDATA[
  1064. $select = $db->select()
  1065. ->from('products');
  1066. $sql = $select->__toString();
  1067. echo "$sql\n";
  1068. // Ausgabe ist dieser String:
  1069. // SELECT * FROM "products"
  1070. ]]></programlisting>
  1071. </example>
  1072. </sect3>
  1073. </sect2>
  1074. <sect2 id="zend.db.select.other">
  1075. <title>Andere Methoden</title>
  1076. <para>
  1077. Dieser Abschnitt beschreibt andere Methoden der <classname>Zend_Db_Select</classname>
  1078. Klasse, welche bisher nicht beschrieben wurden: <methodname>getPart()</methodname> und
  1079. <methodname>reset()</methodname>.
  1080. </para>
  1081. <sect3 id="zend.db.select.other.get-part">
  1082. <title>Abfragen von Teilen des Select-Objekts</title>
  1083. <para>
  1084. Die Methode <methodname>getPart()</methodname> gibt eine Repräsentation eines Teils
  1085. der <acronym>SQL</acronym>-Anfrage zurück. Zum Beispiel kann diese Methode verwendet
  1086. werden um ein Array von Ausdrücke des <constant>WHERE</constant>-Abschnitts, ein
  1087. Array von Spalten (oder Spaltenausdrücke) von <constant>SELECT</constant> oder
  1088. die Werte der Spaltenzahl und Auslassungen des <constant>LIMIT</constant>-Abschnitts
  1089. zu erhalten.
  1090. </para>
  1091. <para>
  1092. Die Rückgabe ist kein String, der ein Fragment der <acronym>SQL</acronym>-Syntax
  1093. enthält. Der Rückgabewert ist eine interne Repräsentation, was typischerweise eine
  1094. Arraystruktur ist, welche Werte und Ausdrücke enthält. Jeder Teil der Anfrage hat
  1095. eine unterschiedliche Struktur.
  1096. </para>
  1097. <para>
  1098. Das einzige Argument der Methode <methodname>getPart()</methodname> ist ein String,
  1099. der den zurückzugebenden Teil der Anfrage bezeichnet. Zum Beispiel bezeichnet der
  1100. String <command>'from'</command> den Teil des Select-Objekts, welcher Informationen
  1101. über den <constant>FROM</constant>-Abschnitt, einschließlich verbundener Tabellen
  1102. enthält.
  1103. </para>
  1104. <para>
  1105. Die Klasse <classname>Zend_Db_Select</classname> definiert Konstanten, die für Teile
  1106. der <acronym>SQL</acronym>-Anfrage verwendet werden können. Es können die
  1107. Konstantendefinitionen oder die literalen Strings verwendet werden.
  1108. </para>
  1109. <table id="zend.db.select.other.get-part.table">
  1110. <title>Konstanten, die von getPart() und reset() verwendet werden</title>
  1111. <tgroup cols="2">
  1112. <thead>
  1113. <row>
  1114. <entry>Konstante</entry>
  1115. <entry>String Wert</entry>
  1116. </row>
  1117. </thead>
  1118. <tbody>
  1119. <row>
  1120. <entry><constant>Zend_Db_Select::DISTINCT</constant></entry>
  1121. <entry><command>'distinct'</command></entry>
  1122. </row>
  1123. <row>
  1124. <entry><constant>Zend_Db_Select::FOR_UPDATE</constant></entry>
  1125. <entry><command>'forupdate'</command></entry>
  1126. </row>
  1127. <row>
  1128. <entry><constant>Zend_Db_Select::COLUMNS</constant></entry>
  1129. <entry><command>'columns'</command></entry>
  1130. </row>
  1131. <row>
  1132. <entry><constant>Zend_Db_Select::FROM</constant></entry>
  1133. <entry><command>'from'</command></entry>
  1134. </row>
  1135. <row>
  1136. <entry><constant>Zend_Db_Select::WHERE</constant></entry>
  1137. <entry><command>'where'</command></entry>
  1138. </row>
  1139. <row>
  1140. <entry><constant>Zend_Db_Select::GROUP</constant></entry>
  1141. <entry><command>'group'</command></entry>
  1142. </row>
  1143. <row>
  1144. <entry><constant>Zend_Db_Select::HAVING</constant></entry>
  1145. <entry><command>'having'</command></entry>
  1146. </row>
  1147. <row>
  1148. <entry><constant>Zend_Db_Select::ORDER</constant></entry>
  1149. <entry><command>'order'</command></entry>
  1150. </row>
  1151. <row>
  1152. <entry><constant>Zend_Db_Select::LIMIT_COUNT</constant></entry>
  1153. <entry><command>'limitcount'</command></entry>
  1154. </row>
  1155. <row>
  1156. <entry><constant>Zend_Db_Select::LIMIT_OFFSET</constant></entry>
  1157. <entry><command>'limitoffset'</command></entry>
  1158. </row>
  1159. </tbody>
  1160. </tgroup>
  1161. </table>
  1162. <example id="zend.db.select.other.get-part.example">
  1163. <title>Beispiel der Methode getPart()</title>
  1164. <programlisting language="php"><![CDATA[
  1165. $select = $db->select()
  1166. ->from('products')
  1167. ->order('product_id');
  1168. // Ein literaler String kann verwendet werden, um den Abschnitt zu definieren
  1169. $orderData = $select->getPart( 'order' );
  1170. // Eine Konstante kann verwendet werden, um denselben Abschnitt zu definieren
  1171. $orderData = $select->getPart( Zend_Db_Select::ORDER );
  1172. // Der zurückgegebene Wert kann eine Arraystruktur sein, kein String.
  1173. // Jeder Abschnitt hat eine unterschiedliche Struktur.
  1174. print_r( $orderData );
  1175. ]]></programlisting>
  1176. </example>
  1177. </sect3>
  1178. <sect3 id="zend.db.select.other.reset">
  1179. <title>Zurücksetzen von Teilen des Select-Objekts</title>
  1180. <para>
  1181. Die Methode <methodname>reset()</methodname> ermöglicht es, einen angegebenen Teil
  1182. der <acronym>SQL</acronym>-Anfrage zu löschen oder, wenn der Parameter ausgelassen
  1183. ist, alle Teile der <acronym>SQL</acronym>-Anfrage zu löschen.
  1184. </para>
  1185. <para>
  1186. Das einzige Argument ist optional. Es kann der Teil der Anfrage angegeben werden,
  1187. der gelöscht werden soll, unter Nutzung des gleichen Strings wie er als Argument
  1188. der Methode <methodname>getPart()</methodname> verwendet wird. Der angegebene Teil
  1189. wird auf einen Standardwert zurück gesetzt.
  1190. </para>
  1191. <para>
  1192. Wenn der Parameter ausgelassen wird, setzt <methodname>reset()</methodname> alle
  1193. geänderten Teile auf einen Standardwert zurück. Dadurch ist das
  1194. <classname>Zend_Db_Select</classname>-Objekt gleichwertig mit einem neuen Objekt,
  1195. wie wenn es gerade instanziiert wurde.
  1196. </para>
  1197. <example id="zend.db.select.other.reset.example">
  1198. <title>Beispiel der Methode reset()</title>
  1199. <programlisting language="php"><![CDATA[
  1200. // Erzeugt diese Anfrage:
  1201. // SELECT p.*
  1202. // FROM "products" AS p
  1203. // ORDER BY "product_name"
  1204. $select = $db->select()
  1205. ->from(array('p' => 'products')
  1206. ->order('product_name');
  1207. // Geänderte Anforderungen, stattdessen sortiert nach einer anderen Spalte:
  1208. // SELECT p.*
  1209. // FROM "products" AS p
  1210. // ORDER BY "product_id"
  1211. // Lösche einen Abschnitt, damit er neu defniert werden kann
  1212. $select->reset( Zend_Db_Select::ORDER );
  1213. // und definiere eine andere Spalte
  1214. $select->order('product_id');
  1215. // Lösche alle Abschnitte von der Abfrage
  1216. $select->reset();
  1217. ]]></programlisting>
  1218. </example>
  1219. </sect3>
  1220. </sect2>
  1221. </sect1>