Zend_Db_Table_Rowset.xml 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 15103 -->
  4. <sect1 id="zend.db.table.rowset">
  5. <title>Zend_Db_Table_Rowset</title>
  6. <sect2 id="zend.db.table.rowset.introduction">
  7. <title>導入</title>
  8. <para>
  9. テーブルクラスに対して <code>find()</code>
  10. あるいは <code>fetchAll()</code> メソッドでクエリを実行すると、
  11. 返される結果は <classname>Zend_Db_Table_Rowset_Abstract</classname> 型のオブジェクトとなります。
  12. 行セットには、<classname>Zend_Db_Table_Row_Abstract</classname> を継承したオブジェクトが含まれます。
  13. 行セットを使用して各行オブジェクトに対して順にアクセスし、
  14. 行のデータを読み込んだり変更したりすることができます。
  15. </para>
  16. </sect2>
  17. <sect2 id="zend.db.table.rowset.fetch">
  18. <title>行セットの取得</title>
  19. <para>
  20. <classname>Zend_Db_Table_Abstract</classname> には <classname>find()</classname> と
  21. <code>fetchAll()</code> というメソッドがあります。
  22. これらはどちらも <classname>Zend_Db_Table_Rowset_Abstract</classname> 型のオブジェクトを返します。
  23. </para>
  24. <example id="zend.db.table.rowset.fetch.example">
  25. <title>行セットの取得の例</title>
  26. <programlisting role="php"><![CDATA[
  27. $bugs = new Bugs();
  28. $rowset = $bugs->fetchAll("bug_status = 'NEW'");
  29. ]]>
  30. </programlisting>
  31. </example>
  32. </sect2>
  33. <sect2 id="zend.db.table.rowset.rows">
  34. <title>行セットからの行の取得</title>
  35. <para>
  36. 通常は、行セットそのものよりもその中に含まれる行のほうが重要になります。
  37. この節では、行セットを構成する行の情報を取得する方法について説明します。
  38. </para>
  39. <para>
  40. 正しい形式のクエリであっても、結果がゼロ行となることがありえます。
  41. たとえば、抽出条件に一致する行がデータベース内に存在しない場合などです。
  42. したがって、行セットオブジェクトの中身の行オブジェクトの個数がゼロになることもあります。
  43. <classname>Zend_Db_Table_Rowset_Abstract</classname> は
  44. <code>Countable</code> インターフェイスを実装しているので、
  45. <code>count()</code> を使用すると行セット内の行の数を調べられます。
  46. </para>
  47. <example id="zend.db.table.rowset.rows.counting.example">
  48. <title>行セット内の行の数を数える</title>
  49. <programlisting role="php"><![CDATA[
  50. $rowset = $bugs->fetchAll("bug_status = 'FIXED'");
  51. $rowCount = count($rowset);
  52. if ($rowCount > 0) {
  53. echo "見つかった行数は $rowCount です";
  54. } else {
  55. echo 'クエリにマッチする行がありません';
  56. }
  57. ]]>
  58. </programlisting>
  59. </example>
  60. <example id="zend.db.table.rowset.rows.current.example">
  61. <title>行セットからの単一の行の読み込み</title>
  62. <para>
  63. 行セットから行にアクセスするための一番簡単な方法は
  64. <code>current()</code> メソッドを使用することです。
  65. これは、行セットに含まれる行数がひとつである場合に最適です。
  66. </para>
  67. <programlisting role="php"><![CDATA[
  68. $bugs = new Bugs();
  69. $rowset = $bugs->fetchAll("bug_id = 1");
  70. $row = $rowset->current();
  71. ]]>
  72. </programlisting>
  73. </example>
  74. <para>
  75. 行セットに含まれる行数がゼロの場合、<code>current()</code>
  76. が返す値は PHP の <code>null</code> 値となります。
  77. </para>
  78. <example id="zend.db.table.rowset.rows.iterate.example">
  79. <title>行セットの順次処理</title>
  80. <para>
  81. <classname>Zend_Db_Table_Rowset_Abstract</classname> を継承したオブジェクトは
  82. <code>SeekableIterator</code> インターフェイスを実装しています。つまり、
  83. <code>foreach</code> ループを使用できるということです。
  84. これを使用して取得した個々の値は <classname>Zend_Db_Table_Row_Abstract</classname>
  85. オブジェクトとなり、これがテーブルの各行に対応します。
  86. </para>
  87. <programlisting role="php"><![CDATA[
  88. $bugs = new Bugs();
  89. // テーブルのすべてのレコードを取得します
  90. $rowset = $bugs->fetchAll();
  91. foreach ($rowset as $row) {
  92. // 出力は 'Zend_Db_Table_Row' あるいはそれに似たものとなります
  93. echo get_class($row) . "\n";
  94. // 行のカラムを読み込みます
  95. $status = $row->bug_status;
  96. // 現在の行のカラムの値を変更します
  97. $row->assigned_to = 'mmouse';
  98. // 変更をデータベースに書き出します
  99. $row->save();
  100. }
  101. ]]>
  102. </programlisting>
  103. </example>
  104. <example id="zend.db.table.rowset.rows.seek.example">
  105. <title>行セット内の既知の位置への移動</title>
  106. <para>
  107. <code>SeekableIterator</code> は、
  108. イテレータ内の特定の位置に移動することができます。
  109. そのために使用するのが <code>seek()</code> メソッドです。
  110. 行番号を渡すと、行セット内のその次の位置に移動することができます。
  111. 行番号は 0 から始まることに注意しましょう。
  112. インデックスが間違っている場合、あるいは存在しない場合は例外がスローされます。
  113. <code>count()</code> を使って結果の行数を確認してから移動するようにしましょう。
  114. </para>
  115. <programlisting role="php"><![CDATA[
  116. $bugs = new Bugs();
  117. // テーブルのすべてのレコードを取得します
  118. $rowset = $bugs->fetchAll();
  119. // イテレータを 9 番目の要素に移動します (最初の要素がゼロです)
  120. $rowset->seek(8);
  121. // それを取得します
  122. $row9 = $rowset->current();
  123. // そして使用します
  124. $row9->assigned_to = 'mmouse';
  125. $row9->save();
  126. ]]>
  127. </programlisting>
  128. </example>
  129. <para>
  130. <code>getRow()</code> は、位置がわかっている場合に
  131. 行セット内の特定の行を取得するためのメソッドです。
  132. しかし、この位置はゼロから数え始めることを忘れないようにしましょう。
  133. <code>getRow()</code> の最初のパラメータは、位置を表す整数値です。
  134. 2 番目のパラメータはオプションで、boolean 値を指定します。
  135. これは、Rowset イテレータも同時にその場所に移動させるのかどうかを表します
  136. (デフォルトは false です)。このメソッドはデフォルトでは
  137. <classname>Zend_Db_Table_Row</classname> オブジェクトを返します。
  138. 指定した位置が存在しない場合は例外をスローします。
  139. 例を示します。
  140. </para>
  141. <programlisting role="php"><![CDATA[
  142. $bugs = new Bugs();
  143. // テーブルのすべてのレコードを取得します
  144. $rowset = $bugs->fetchAll();
  145. // 9 番目の要素を取得します
  146. $row9->getRow(8);
  147. // そして使用します
  148. $row9->assigned_to = 'mmouse';
  149. $row9->save();
  150. ]]>
  151. </programlisting>
  152. <para>
  153. 個々の行オブジェクトにアクセスすると、後は
  154. <xref linkend="zend.db.table.row" /> で説明しているメソッド群を用いて行を操作できます。
  155. </para>
  156. </sect2>
  157. <sect2 id="zend.db.table.rowset.to-array">
  158. <title>行セットの配列としての取得</title>
  159. <para>
  160. 行セット内のすべてのデータに対して配列としてアクセスするには、
  161. 行セットオブジェクトの <code>toArray()</code> メソッドを使用します。
  162. これは、各行単位でひとつの要素となる配列を返します。
  163. 各エントリは連想配列となり、カラム名とその値が関連付けられています。
  164. </para>
  165. <example id="zend.db.table.rowset.to-array.example">
  166. <title>toArray() の使用法</title>
  167. <programlisting role="php"><![CDATA[
  168. $bugs = new Bugs();
  169. $rowset = $bugs->fetchAll();
  170. $rowsetArray = $rowset->toArray();
  171. $rowCount = 1;
  172. foreach ($rowsetArray as $rowArray) {
  173. echo "row #$rowCount:\n";
  174. foreach ($rowArray as $column => $value) {
  175. echo "\t$column => $value\n";
  176. }
  177. ++$rowCount;
  178. echo "\n";
  179. }
  180. ]]>
  181. </programlisting>
  182. </example>
  183. <para>
  184. <code>toArray()</code> が返す配列は、更新できません。
  185. つまり、配列内の値を変更することは可能ですが、
  186. それをデータベースに反映させることはできません。
  187. </para>
  188. </sect2>
  189. <sect2 id="zend.db.table.rowset.serialize">
  190. <title>行セットのシリアライズと復元</title>
  191. <para>
  192. <classname>Zend_Db_Table_Rowset_Abstract</classname> 型のオブジェクトはシリアライズ可能です。
  193. 個別の行オブジェクトをシリアライズするのと同じような方式で、
  194. 行セットをシリアライズして後ほどそれを復元することができます。
  195. </para>
  196. <example id="zend.db.table.rowset.serialize.example.serialize">
  197. <title>行セットのシリアライズ</title>
  198. <para>
  199. PHP の <code>serialize()</code> 関数を使用して、
  200. 行セットオブジェクトのバイトストリームを含む文字列を作成します。
  201. </para>
  202. <programlisting role="php"><![CDATA[
  203. $bugs = new Bugs();
  204. $rowset = $bugs->fetchAll();
  205. // オブジェクトをシリアライズします
  206. $serializedRowset = serialize($rowset);
  207. // これで、$serializedRowset をファイルなどに書き出すことができます
  208. ]]>
  209. </programlisting>
  210. </example>
  211. <example id="zend.db.table.rowset.serialize.example.unserialize">
  212. <title>シリアライズした行セットの復元</title>
  213. <para>
  214. PHP の <code>unserialize()</code> 関数を使用して、
  215. オブジェクトのバイトストリームを含む文字列を復元します。
  216. この関数は、もとのオブジェクトを返します。
  217. </para>
  218. <para>
  219. 返された行セットオブジェクトは、
  220. <emphasis>接続が切断された</emphasis> 状態であることに注意しましょう。
  221. 行セットオブジェクトやその内部の行オブジェクト、そしてそのプロパティを読み込むことはできますが、
  222. その値を変更することはできません。また、データベース接続を必要とするようなメソッド
  223. (たとえば従属テーブルに対するクエリなど) も実行できません。
  224. </para>
  225. <programlisting role="php"><![CDATA[
  226. $rowsetDisconnected = unserialize($serializedRowset);
  227. // これでオブジェクトのプロパティを使用できますが、読み込み専用です
  228. $row = $rowsetDisconnected->current();
  229. echo $row->bug_description;
  230. ]]>
  231. </programlisting>
  232. </example>
  233. <note>
  234. <title>復元した行セットは、なぜ切断された状態なのですか?</title>
  235. <para>
  236. シリアライズしたオブジェクトは、可読形式の文字列となります。
  237. データベースのアカウントやパスワードといった情報を
  238. 暗号化せずにプレーンテキストにシリアライズして保存すると、
  239. セキュリティ上問題となります。
  240. そのようなデータを無防備な状態でテキストファイルに保存したりしたくはないでしょう。
  241. またメールなどで攻撃者に覗き見られることも好まないはずです。
  242. シリアライズされたオブジェクトは、
  243. 正しい認証情報を知らない限りデータベースにアクセスすることはできません。
  244. </para>
  245. </note>
  246. <para>
  247. 切断された行セットの接続を復活させるには、
  248. <code>setTable()</code> メソッドを使用します。このメソッドへの引数としては、
  249. <classname>Zend_Db_Table_Abstract</classname> 型のオブジェクトを作成して渡します。
  250. テーブルオブジェクトを作成するには、データベースとの接続が必要です。
  251. そのテーブルと行セットを関連付けることで、行セットがデータベースにアクセスできるようになります。
  252. それ以降は、行オブジェクトの値を変更してデータベースに保存できるようになります。
  253. </para>
  254. <example id="zend.db.table.rowset.serialize.example.set-table">
  255. <title>生きたデータとしての行セットの復活</title>
  256. <programlisting role="php"><![CDATA[
  257. $rowset = unserialize($serializedRowset);
  258. $bugs = new Bugs();
  259. // この行セットをテーブルに再接続し、
  260. // データベースとの接続を復活させます
  261. $rowset->setTable($bugs);
  262. $row = $rowset->current();
  263. // これで、行の内容を変更して保存することができます
  264. $row->bug_status = 'FIXED';
  265. $row->save();
  266. ]]>
  267. </programlisting>
  268. </example>
  269. <para>
  270. 行セットを <code>setTable()</code> で復活させると、
  271. その中に含まれる行オブジェクトもすべて復活した状態になります。
  272. </para>
  273. </sect2>
  274. <sect2 id="zend.db.table.rowset.extending">
  275. <title>行セットクラスの拡張</title>
  276. <para>
  277. <classname>Zend_Db_Table_Rowset_Abstract</classname> を継承した新たな具象クラスを作成し、
  278. それを用いて行セットのインスタンスを作成することができます。
  279. 独自の行クラスを指定するには、テーブルクラスの protected
  280. メンバである <code>$_rowsetClass</code> を使用するか、
  281. テーブルオブジェクトのコンストラクタの引数の配列で指定します。
  282. </para>
  283. <example id="zend.db.table.rowset.extending.example">
  284. <title>独自の行セットクラスの指定</title>
  285. <programlisting role="php"><![CDATA[
  286. class MyRowset extends Zend_Db_Table_Rowset_Abstract
  287. {
  288. // ...独自の処理
  289. }
  290. // 独自の行セットを、テーブルクラスの全インスタンスで
  291. // デフォルトとして使用するように設定します
  292. class Products extends Zend_Db_Table_Abstract
  293. {
  294. protected $_name = 'products';
  295. protected $_rowsetClass = 'MyRowset';
  296. }
  297. // あるいは、テーブルクラスの特定のインスタンスでのみ
  298. // 独自の行セットクラスを使用するように設定します
  299. $bugs = new Bugs(array('rowsetClass' => 'MyRowset'));
  300. ]]>
  301. </programlisting>
  302. </example>
  303. <para>
  304. 一般的には、標準の具象クラス <classname>Zend_Db_Rowset</classname>
  305. でたいていの場合は十分でしょう。しかし、
  306. 特定のテーブルに固有の処理を行セットに追加したくなることもあります。
  307. たとえば、行セット内のすべての行の内容の集計用のメソッドなどです。
  308. </para>
  309. <example id="zend.db.table.rowset.extending.example-aggregate">
  310. <title>行セットクラスに新しいメソッドを追加する例</title>
  311. <programlisting role="php"><![CDATA[
  312. class MyBugsRowset extends Zend_Db_Table_Rowset_Abstract
  313. {
  314. /**
  315. * 現在の行セットのなかで、'updated_at' カラムの値が
  316. * 最大である行を見つけます
  317. */
  318. public function getLatestUpdatedRow()
  319. {
  320. $max_updated_at = 0;
  321. $latestRow = null;
  322. foreach ($this as $row) {
  323. if ($row->updated_at > $max_updated_at) {
  324. $latestRow = $row;
  325. }
  326. }
  327. return $latestRow;
  328. }
  329. }
  330. class Bugs extends Zend_Db_Table_Abstract
  331. {
  332. protected $_name = 'bugs';
  333. protected $_rowsetClass = 'MyBugsRowset';
  334. }
  335. ]]>
  336. </programlisting>
  337. </example>
  338. </sect2>
  339. </sect1>
  340. <!--
  341. vim:se ts=4 sw=4 et:
  342. -->