Zend_Db_Table_Rowset.xml 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- EN-Revision: 15156 -->
  3. <!-- Reviewed: no -->
  4. <sect1 id="zend.db.table.rowset">
  5. <title>Zend_Db_Table_Rowset</title>
  6. <sect2 id="zend.db.table.rowset.introduction">
  7. <title>Einführung</title>
  8. <para>
  9. Wenn eine Datenbankabfrage über eine Tabellenklasse ausgeführt wird, genauer über deren Methoden
  10. <code>find()</code> und <code>fetchAll()</code>, wird das Ergebnis als Objekt vom Typ
  11. <classname>Zend_Db_Table_Rowset_Abstract</classname> zurückgegeben. Ein Zeilensatz enthält eine Sammlung von
  12. <classname>Zend_Db_Table_Row_Abstract</classname>-Objekten. Das Zeilensatz-Objekt implementiert das
  13. <code>Iterator</code>-Interface, sodass es auch einfach via <code>foreach</code> durchgegangen
  14. werden kann und Lese- und Schreibzugriff auf die einzelnen Zeilen möglich ist.
  15. </para>
  16. </sect2>
  17. <sect2 id="zend.db.table.rowset.fetch">
  18. <title>Einen Zeilensatz lesen</title>
  19. <para>
  20. <classname>Zend_Db_Table_Abstract</classname> bietet die Methoden <code>find()</code> und <code>fetchAll()</code>,
  21. die beide ein Objekt vom Typ <classname>Zend_Db_Table_Rowset_Abstract</classname> zurückgeben.
  22. </para>
  23. <example id="zend.db.table.rowset.fetch.example">
  24. <title>Einen Zeilensatz lesen</title>
  25. <programlisting role="php"><![CDATA[
  26. $bugs = new Bugs();
  27. $rowset = $bugs->fetchAll("bug_status = 'NEW'");
  28. ]]></programlisting>
  29. </example>
  30. </sect2>
  31. <sect2 id="zend.db.table.rowset.rows">
  32. <title>Zeilen aus einem Zeilensatz auslesen</title>
  33. <para>
  34. Der Zeilensatz selber ist normalerweise weniger interessant als die Zeilen, die er enthält.
  35. Dieser Abschnitt zeigt, wie die Zeilen, die im Zeilensatz enthalten sind, auslesbar sind.
  36. </para>
  37. <para>
  38. Eine normale Abfrage gibt null Zeilen zurück, wenn keine Zeilen in der Datenbank die Bedingungen der
  39. Abfrage erfüllt. Daher kann eine Zeilensatz-Objekt auch null Zeilenobjekte enthalten.
  40. Weil <classname>Zend_Db_Table_Rowset_Abstract</classname> auch das Interface <code>Countable</code>
  41. (dt.: Zählbar) implementiert, kann die Funktion count() genutzt werden, um die Anzahl der Zeilen im
  42. Zeilensatz zu erhalten.
  43. </para>
  44. <example id="zend.db.table.rowset.rows.counting.example">
  45. <title>Zeilen in einem Zeilensatz zählen</title>
  46. <programlisting role="php"><![CDATA[
  47. $rowset = $bugs->fetchAll("bug_status = 'FIXED'");
  48. $rowCount = count($rowset);
  49. if ($rowCount > 0) {
  50. echo "$rowCount Zeilen gefunden!";
  51. } else {
  52. echo 'keine Zeilen für die Abfrage gefunden.';
  53. }
  54. ]]></programlisting>
  55. </example>
  56. <example id="zend.db.table.rowset.rows.current.example">
  57. <title>Eine einzelne Zeile aus einem Zeilensatz auslesen</title>
  58. <para>
  59. Die einfachste Art, eine Zeile aus einem Zeilensatz auszulesen, ist die Methode
  60. <code>current()</code>. Diese ist vor allem dann nützlich, wenn der Zeilensatz genau eine Zeile
  61. enthält.
  62. </para>
  63. <programlisting role="php"><![CDATA[
  64. $bugs = new Bugs();
  65. $rowset = $bugs->fetchAll("bug_id = 1");
  66. $row = $rowset->current();
  67. ]]></programlisting>
  68. </example>
  69. <para>
  70. Wenn der Zeilensatz keine Zeilen enthält, gibt <code>current()</code> den Wert <code>null</code> zurück.
  71. </para>
  72. <example id="zend.db.table.rowset.rows.iterate.example">
  73. <title>Einen Zeilensatz durchlaufen</title>
  74. <para>
  75. Objekte, die von <classname>Zend_Db_Table_Rowset_Abstract</classname> abstammen,
  76. implementieren das <code>SeekableIterator</code> Interface, was bedeutet, dass es
  77. mit <code>foreach</code> durchlaufen werden kann. Jeder Wert, der auf diesem Weg
  78. zurückgegeben wird, ist ein <classname>Zend_Db_Table_Row_Abstract</classname>-Objekt, das zu
  79. einem Eintrag in der Tabelle gehört.
  80. </para>
  81. <programlisting role="php"><![CDATA[
  82. $bugs = new Bugs();
  83. // Alle Zeilen aus der Tabelle lesen
  84. $rowset = $bugs->fetchAll();
  85. foreach ($rowset as $row) {
  86. // Ausgabe: 'Zend_Db_Table_Row' oder ähnlich,
  87. // je nach benutzter Zeilenklasse
  88. echo get_class($row) . "\n";
  89. // Spalte einer Zeile auslesen
  90. $status = $row->bug_status;
  91. // eine Spalte der aktuellen Zeile modifizieren
  92. $row->assigned_to = 'mmouse';
  93. // Änderung in der Datenbank speichern
  94. $row->save();
  95. }
  96. ]]></programlisting>
  97. </example>
  98. <example id="zend.db.table.rowset.rows.seek.example">
  99. <title>Eine bekannte Position in einem Rowset suchen</title>
  100. <para>
  101. <code>SeekableIterator</code> erlaubt es eine Position zu suchen auf die der
  102. Iterator springen soll. Hierfür kann einfach die <code>seek()</code> Methode
  103. verwendet werden. Es kann ein Integer übergeben werden die der Nummer der Zeile
  104. repräsentiert auf die das Rowset als nächstes zeigen soll, wobei man nicht
  105. vergessen sollte das der Index mit 0 beginnt. Wenn der Index falsch ist, z.b. nicht
  106. existiert, wird eine Ausnahme geworfen. Man sollte <code>count()</code>
  107. verwenden um die Anzahl an Ergebnissen zu prüfen bevor eine Position gesucht wird.
  108. </para>
  109. <programlisting role="php"><![CDATA[
  110. $bugs = new Bugs();
  111. // Alle Einträge von der Tabelle holen
  112. $rowset = $bugs->fetchAll();
  113. // Den Iterator zum 9ten Element bringen (null ist das erste Element) :
  114. $rowset->seek(8);
  115. // es empfangen
  116. $row9 = $rowset->current();
  117. // und es verwenden
  118. $row9->assigned_to = 'mmouse';
  119. $row9->save();
  120. ]]></programlisting>
  121. </example>
  122. <para>
  123. <code>getRow()</code> erlaubt es eine spezielle Zeile im Rowset zu erhalten wenn
  124. dessen Position bekannt ist; trotzdem sollte nicht vergessen werden dass die
  125. Position mit dem Index null beginnt. Der erste Parameter für <code>getRow()</code>
  126. ist ein Integer für die gewünschte Position. Der zweite optionale Parameter ist ein
  127. Boolean; Es teilt dem Rowset Iterator mit ob er zur gleichen Zeit diese Position
  128. suchen muss, oder nicht (standard ist nicht). Diese Methode gibt standardmäßig ein
  129. <classname>Zend_Db_Table_Row</classname> Objekt zurück. Wenn die angefragte Position nicht existiert wird
  130. eine Ausnahme geworfen. Hier ist ein Beispiel:
  131. </para>
  132. <programlisting role="php"><![CDATA[
  133. $bugs = new Bugs();
  134. // Alle Einträge von der Tabelle holen
  135. $rowset = $bugs->fetchAll();
  136. // Sofort das 9te Element holen:
  137. $row9->getRow(8);
  138. // und es verwenden:
  139. $row9->assigned_to = 'mmouse';
  140. $row9->save();
  141. ]]></programlisting>
  142. <para>
  143. Sobald der Zugriff auf ein Zeilenobjekt besteht, kann dieses mit den Methoden manipuliert werden,
  144. die in <xref linkend="zend.db.table.row" /> beschrieben werden.
  145. </para>
  146. </sect2>
  147. <sect2 id="zend.db.table.rowset.to-array">
  148. <title>Einen Zeilensatz als Array lesen</title>
  149. <para>
  150. Auf die gesamten Daten in einem Zeilensatz kann mithilfe der Methode <code>toArray()</code> des
  151. Zeilensatz-Objekts auch als Array zugegriffen werden. Diese Methode gibt ein Array mit einem Eintrag
  152. je Zeile zurück. Jeder dieser Einträge ist ein assoziatives Array mit Spaltennamen als Schlüsseln und
  153. deren Daten als Werten.
  154. </para>
  155. <example id="zend.db.table.rowset.to-array.example">
  156. <title>Benutzung von toArray()</title>
  157. <programlisting role="php"><![CDATA[
  158. $bugs = new Bugs();
  159. $rowset = $bugs->fetchAll();
  160. $rowsetArray = $rowset->toArray();
  161. $rowCount = 1;
  162. foreach ($rowsetArray as $rowArray) {
  163. echo "Zeile #$rowCount:\n";
  164. foreach ($rowArray as $column => $value) {
  165. echo "\t$column => $value\n";
  166. }
  167. ++$rowCount;
  168. echo "\n";
  169. }
  170. ]]></programlisting>
  171. </example>
  172. <para>
  173. Das Array, das von <code>toArray()</code>zurückgegeben wird, ist nicht update-fähig.
  174. Die Werte des Arrays können wie bei jedem Array modifiziert werden, aber Änderungen an diesem Array
  175. werden nicht direkt in der Datenbank gespeichert.
  176. </para>
  177. </sect2>
  178. <sect2 id="zend.db.table.rowset.serialize">
  179. <title>Einen Zeilensatz serialisieren / deserialisieren</title>
  180. <para>
  181. Objekte vom Typ <classname>Zend_Db_Table_Rowset_Abstract</classname> sind serialisierbar auf eine ähnliche
  182. Art, wie auch einzelne Zeilen-Objekte serialisierbar und deserialisierbar sind.
  183. </para>
  184. <example id="zend.db.table.rowset.serialize.example.serialize">
  185. <title>Einen Zeilensatz serialisieren</title>
  186. <para>
  187. PHPs <code>serialize()</code>-Funktion wird genutzt, um einen Byte-Stream zu erzeugen.
  188. Dieser repräsentiert das Zeilensatz-Objekt.
  189. </para>
  190. <programlisting role="php"><![CDATA[
  191. $bugs = new Bugs();
  192. $rowset = $bugs->fetchAll();
  193. // Objekt serialisieren
  194. $serializedRowset = serialize($rowset);
  195. // Jetzt kann $serializedRowset bspw.
  196. // in einer Datei gespeichert werden
  197. ]]></programlisting>
  198. </example>
  199. <example id="zend.db.table.rowset.serialize.example.unserialize">
  200. <title>Einen Zeilensatz deserialisieren</title>
  201. <para>
  202. PHPs <code>unserialize()</code> stellt aus einer Zeichenkette mit einem Byte-Stream ein Objekt wieder
  203. her. Die Funktion gibt das originale Objekt zurück.
  204. </para>
  205. <para>
  206. Bitte beachten: Das zurückgegebene Zeilensatz-Objekt ist
  207. <emphasis>nicht mit der Datenbank verbunden</emphasis>.
  208. Das Zeilensatz-Objekt kann durchlaufen werden und die Zeilenobjekte können gelesen werden,
  209. aber es können keine Zeilenwerte verändert oder andere Operationen ausgeführt werden,
  210. die eine Datenbankverbindung benötigen (beispielsweise Abfragen auf verwandte Tabellen).
  211. </para>
  212. <programlisting role="php"><![CDATA[
  213. $rowsetDisconnected = unserialize($serializedRowset);
  214. // Jetzt können Objekt-Methoden und -Eigenschaften genutzt werden,
  215. // aber nur lesend.
  216. $row = $rowsetDisconnected->current();
  217. echo $row->bug_description;
  218. ]]></programlisting>
  219. </example>
  220. <note>
  221. <title>Warum werden Zeilensatz-Objekte unverbunden deserialisiert?</title>
  222. <para>
  223. Ein serialisiertes Objekt ist eine Zeichenkette, die lesbar für jeden ist, dem sie vorliegt.
  224. Es könnte ein Sicherheitsrisiko sein, Parameter wie Datenbank-Loginname und -Passwort
  225. in simplem, unverschlüsseltem Text abzulegen.
  226. Es ist nicht wünschenswert, solche Daten in einer Textdatei abzulegen, die nicht geschützt ist,
  227. oder sie in einer E-Mail oder einem anderen Medium zu versenden, das leicht von potentiellen
  228. Angreifern lesbar ist.
  229. Der Leser des serialisierten Objekts sollte es nicht benutzen können, um Zugriff zur Datenbank
  230. zu erhalten, ohne richtige Logindaten zu kennen.
  231. </para>
  232. </note>
  233. <para>
  234. Ein nicht verbundenes Zeilensatz-Objekt kann mithilfe der Methode <code>setTable()</code>
  235. reaktiviert werden. Das Argument dieser Methode ist ein valides
  236. <classname>Zend_Db_Table_Abstract</classname>-Objekt,
  237. das vom Benutzer erstellt wird. Für das Erstellen eines Tabellenobjekts wird eine aktive
  238. Datenbankverbindung benötigt, also wird, indem die Tabelle wieder mit dem Zeilenobjekt verknüpft
  239. wird, auch der Datenbankzugriff wiederhergestellt. Ab diesem Zeitpunkt können Werte in den enthaltenen
  240. Zeilenobjekten wieder verändert und in der Datenbank gespeichert werden.
  241. </para>
  242. <example id="zend.db.table.rowset.serialize.example.set-table">
  243. <title>Einen Zeilensatz als Live-Daten reaktivieren</title>
  244. <programlisting role="php"><![CDATA[
  245. $rowset = unserialize($serializedRowset);
  246. $bugs = new Bugs();
  247. // Den Zeilensatz wieder mit einer Tabelle
  248. // und damit mit einer aktiven Datenbankverbindung verknüpfen
  249. $rowset->setTable($bugs);
  250. $row = $rowset->current();
  251. // Jetzt können wieder Werte geändert und danach gespeichert werden
  252. $row->bug_status = 'FIXED';
  253. $row->save();
  254. ]]></programlisting>
  255. </example>
  256. <para>
  257. Wenn ein Zeilensatz mit <code>setTable()</code> reaktiviert wird,
  258. reaktiviert das auch alle enthaltenen Zeilen-Objekte.
  259. </para>
  260. </sect2>
  261. <sect2 id="zend.db.table.rowset.extending">
  262. <title>Die Zeilensatz-Klasse erweitern</title>
  263. <para>
  264. Es können auch alternative Klassen für Zeilensätze benutzt werden, wenn diese
  265. <classname>Zend_Db_Table_Rowset_Abstract</classname> erweitern. Der Name der eigenen Zeilensatz-Klasse wird entweder
  266. in der geschützten Tabellenklassen-Eigenschaft <code>$_rowsetClass</code> oder als Teil des
  267. Array-Arguments des Konstruktors eines Tabellenobjekts übergeben.
  268. </para>
  269. <example id="zend.db.table.rowset.extending.example">
  270. <title>Eine eigene Zeilensatz-Klasse angeben</title>
  271. <programlisting role="php"><![CDATA[
  272. class MyRowset extends Zend_Db_Table_Rowset_Abstract
  273. {
  274. // ...Anpassungen
  275. }
  276. // Eine eigene Zeilensatz-Klasse angeben, die standardmäßig
  277. // in allen Instanzen der Tabellenklasse benutzt wird.
  278. class Products extends Zend_Db_Table_Abstract
  279. {
  280. protected $_name = 'products';
  281. protected $_rowsetClass = 'MyRowset';
  282. }
  283. // Oder eine eigene Zeilensatz-Klasse angeben, die in einer
  284. // Instanz einer Tabellenklasse benutzt wird
  285. $bugs = new Bugs(array('rowsetClass' => 'MyRowset'));
  286. ]]></programlisting>
  287. </example>
  288. <para>
  289. Typischerweise reicht die Standardklasse <classname>Zend_Db_Rowset</classname> für die meisten Benutzungsfälle aus.
  290. Trotzdem könnte es nützlich sein, neue Logik in einen Zeilensatz einzubauen, die für eine
  291. bestimmte Tabelle nötig ist. Beispielsweise könnte eine neue Methode einen Durchschnitt aller
  292. Zeilen im Zeilensatz errechnen.
  293. </para>
  294. <example id="zend.db.table.rowset.extending.example-aggregate">
  295. <title>Eine Zeilensatz-Klasse mit einer neuen Methode</title>
  296. <programlisting role="php"><![CDATA[
  297. class MyBugsRowset extends Zend_Db_Table_Rowset_Abstract
  298. {
  299. /**
  300. * Suche nach der Zeile im Zeilensatz, deren
  301. * 'updated_at'-Spalte den größten Wert hat.
  302. */
  303. public function getLatestUpdatedRow()
  304. {
  305. $max_updated_at = 0;
  306. $latestRow = null;
  307. foreach ($this as $row) {
  308. if ($row->updated_at > $max_updated_at) {
  309. $latestRow = $row;
  310. }
  311. }
  312. return $latestRow;
  313. }
  314. }
  315. class Bugs extends Zend_Db_Table_Abstract
  316. {
  317. protected $_name = 'bugs';
  318. protected $_rowsetClass = 'MyBugsRowset';
  319. }
  320. ]]></programlisting>
  321. </example>
  322. </sect2>
  323. </sect1>
  324. <!--
  325. vim:se ts=4 sw=4 et:
  326. -->