File: Zend_Db_Table_Row.xml

package info (click to toggle)
zendframework 1.12.9%2Bdfsg-2
  • links: PTS, VCS
  • area: main
  • in suites: jessie-kfreebsd
  • size: 133,584 kB
  • sloc: xml: 1,311,829; php: 570,173; sh: 170; makefile: 125; sql: 121
file content (619 lines) | stat: -rw-r--r-- 23,886 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
<?xml version="1.0" encoding="UTF-8"?>
<!-- EN-Revision: 24249 -->
<!-- Reviewed: no -->
<sect1 id="zend.db.table.row">
    <title>Zend_Db_Table_Row</title>

    <sect2 id="zend.db.table.row.introduction">
        <title>Introduction</title>

        <para>
            <classname>Zend_Db_Table_Row</classname> est la classe qui donne accès à chacun
            des résultats issus d'un objet <classname>Zend_Db_Table</classname>. Lorsque vous
            exécutez une requête via une classe de Table, alors les résultats sont des objets
            <classname>Zend_Db_Table_Row</classname>. Vous pouvez aussi utiliser ces objets comme
            résultats vides : pour créer des nouveaux résultats à ajouter à la base de
            données.
        </para>

        <para>
            <classname>Zend_Db_Table_Row</classname> est une implémentation du design pattern
            <ulink url="http://www.martinfowler.com/eaaCatalog/rowDataGateway.html">Row Data
            Gateway</ulink>
        </para>
    </sect2>

    <sect2 id="zend.db.table.row.read">
        <title>Récupérer un résultat (un "Row")</title>

        <para>
            <classname>Zend_Db_Table_Abstract</classname> possède des méthodes
            <methodname>find()</methodname> et <methodname>fetchAll()</methodname>, qui retournent un objet de type
            <classname>Zend_Db_Table_Rowset</classname>, et une méthode <methodname>fetchRow()</methodname>, qui
            retourne un objet de type <classname>Zend_Db_Table_Row</classname>.
        </para>

        <example id="zend.db.table.row.read.example">
            <title>Exemple de récupération d'un Row</title>

            <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$row = $bugs->fetchRow($bugs->select()
                            ->where('bug_id = ?', 1));
]]></programlisting>
        </example>

        <para>
            Un objet <classname>Zend_Db_Table_Rowset</classname> contient une collection
            d'objets <classname>Zend_Db_Table_Row</classname>. Voyez <xref
            linkend="zend.db.table.rowset" />.
        </para>

        <example id="zend.db.table.row.read.example-rowset">
            <title>Exemple de lecture d'un Row dans un Rowset</title>

            <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$rowset = $bugs->fetchAll($bugs->select()
                               ->where('bug_status = ?', 1));
$row = $rowset->current();
]]></programlisting>
        </example>

        <sect3 id="zend.db.table.row.read.get">
            <title>Lecture des valeurs des colonnes, dans un Row</title>

            <para>
                <classname>Zend_Db_Table_Row_Abstract</classname> possède des accesseurs. Les
                colonnes <acronym>SQL</acronym> du résultat sont disponibles en lecture et écriture, via des propriétés
                de classe.
            </para>

            <example id="zend.db.table.row.read.get.example">
                <title>Lecture d'une colonne dans un Row</title>

                <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$row = $bugs->fetchRow($bugs->select()
                            ->where('bug_id = ?', 1));

// Affiche la valeur de la colonne bug_description
echo $row->bug_description;
]]></programlisting>
            </example>

            <note>
                <para>
                    Les versions antérieures de <classname>Zend_Db_Table_Row</classname>
                    utilisaient un processus de transformation nommé <emphasis>inflexion</emphasis>
                    pour récupérer les valeurs des colonnes dans un résultat.
                </para>

                <para>
                    Actuellement, <classname>Zend_Db_Table_Row</classname> n'utilise pas
                    d'inflexion. Les noms des propriétés de l'objet doivent correspondre à
                    l'orthographe des noms des colonnes dans la base de données sous-jacente
                </para>
            </note>
        </sect3>

        <sect3 id="zend.db.table.row.read.to-array">
            <title>Récupérer les valeurs des colonnes comme un tableau</title>

            <para>
                Vous pouvez accéder aux données d'un row sous forme de tableau grâce à la
                méthode <methodname>toArray()</methodname>. Celle-ci retourne un tableau associatif.
            </para>

            <example id="zend.db.table.row.read.to-array.example">
                <title>Exemple avec toArray()</title>

                <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$row = $bugs->fetchRow($bugs->select()
                            ->where('bug_id = ?', 1));

// Récupère un tableau associatif column/value
$rowArray = $row->toArray();

// Utilisation comme un tableau normal
foreach ($rowArray as $column => $value) {
    echo "Column: $column\n";
    echo "Value:  $value\n";
}
]]></programlisting>
            </example>

            <para>
                Le tableau retourné par <methodname>toArray()</methodname> n'est pas une référence. Vous
                pouvez modifier ses valeurs, cela n'aura aucune répercussion dans la base de
                données.
            </para>
        </sect3>

        <sect3 id="zend.db.table.row.read.relationships">
            <title>Récupérer des données des tables liées</title>

            <para>
                <classname>Zend_Db_Table_Row_Abstract</classname> possède des méthodes
                permettant de récupérer des données des tables liées à la table interrogée. Voyez
                <xref linkend="zend.db.table.relationships" /> pour plus d'informations sur les
                relations entre les tables.
            </para>
        </sect3>
    </sect2>

    <sect2 id="zend.db.table.row.write">
        <title>Sauvegarde un Row en base de données</title>

        <sect3 id="zend.db.table.row.write.set">
            <title>Changement des valeurs des colonnes d'un Row</title>

            <para>
                Vous pouvez changer les valeurs de chaque colonne du résultat Row, simplement
                avec les accesseurs, comme en lecture. Effectuez une banale affectation.
            </para>

            <para>
                Utiliser l'accesseur pour spécifier une valeur à une colonne d'un résultat Row
                ne répercute pas le comportement immédiatement en base de données. Vous devez
                utiliser explicitement la méthode <methodname>save()</methodname> pour ceci.
            </para>

            <example id="zend.db.table.row.write.set.example">
                <title>Exemple de changement de la valeur d'une colonne dans un Row</title>

                <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$row = $bugs->fetchRow($bugs->select()
                            ->where('bug_id = ?', 1));

// Change la valeur d'une ou plusieurs colonnes
$row->bug_status = 'FIXED';

// MET A JOUR l'enregistrement dans la base de données
$row->save();
]]></programlisting>
            </example>
        </sect3>

        <sect3 id="zend.db.table.row.write.insert">
            <title>Créer un Row vierge</title>

            <para>
                Vous pouvez créer un nouvel enregistrement vierge (Row) pour une table avec la
                méthode <methodname>createRow()</methodname> issue de la classe de cette Table. Vous pouvez
                alors affecter des valeurs à ses colonnes grâce aux accesseurs, comme déjà vu, puis
                enregistrer le Row en base de données avec sa méthode <methodname>save()</methodname>.
            </para>

            <example id="zend.db.table.row.write.insert.example">
                <title>Exemple de création d'un Row vierge pour une table</title>

                <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$newRow = $bugs->createRow();

// affecte des valeurs aux colonnes
$newRow->bug_description = '...description...';
$newRow->bug_status = 'NEW';

// INSERT le nouvel enregistrement dans la base de données
$newRow->save();
]]></programlisting>
            </example>

            <para>
                L'argument optionnel de <methodname>createRow()</methodname> est un tableau associatif qui
                sert à peupler tout de suite l'objet de valeurs.
            </para>

            <example id="zend.db.table.row.write.insert.example2">
                <title>Exemple de remplissage des valeurs d'un nouveau Row vierge</title>

                <programlisting language="php"><![CDATA[
$data = array(
    'bug_description' => '...description...',
    'bug_status'      => 'NEW'
);

$bugs = new Bugs();
$newRow = $bugs->createRow($data);

// INSERT l'enregistrement en base de données
$newRow->save();
]]></programlisting>
            </example>

            <note>
                <para>
                    La méthode <methodname>createRow()</methodname> était nommée <methodname>fetchNew()</methodname>
                    dans les anciennes version de <classname>Zend_Db_Table</classname>. Il est
                    recommandé de ne plus utiliser cette ancienne appellation, même si celle-ci
                    fonctionne toujours actuellement.
                </para>
            </note>
        </sect3>

        <sect3 id="zend.db.table.row.write.set-from-array">
            <title>Changement en masse des valeurs dans un Row</title>

            <para>
                <classname>Zend_Db_Table_Row_Abstract</classname> possède une méthode
                <methodname>setFromArray()</methodname> qui permet de lui peupler ses valeurs avec celles issues
                d'un tableau associatif nom de la colonne / valeur.
            </para>

            <example id="zend.db.table.row.write.set-from-array.example">
                <title>
                    Exemple d'utilisation de setFromArray() avec un enregistrement (Row) vierge
                </title>

                <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$newRow = $bugs->createRow();

// Les données sont dans un tableau associatif
$data = array(
    'bug_description' => '...description...',
    'bug_status'      => 'NEW'
);

// Affecte toutes les valeurs des colonnes en une seule fois
$newRow->setFromArray($data);

// INSERT l'enregistrement en base de données
$newRow->save();
]]></programlisting>
            </example>
        </sect3>

        <sect3 id="zend.db.table.row.write.delete">
            <title>Supprimer un Row</title>

            <para>
                Vous pouvez appeler la méthode <methodname>delete()</methodname> d'un objet Row. Ceci
                supprime les lignes dans la base de données qui correspondent à la clé primaire de
                l'objet Row.
            </para>

            <example id="zend.db.table.row.write.delete.example">
                <title>Effacer un Row</title>

                <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$row = $bugs->fetchRow('bug_id = 1');

// EFFACE cet enregistrement de la base de données
$row->delete();
]]></programlisting>
            </example>

            <para>
                Notez qu'il n'est pas nécessaire d'appeler <methodname>save()</methodname> pour un
                effacement. Celui-ci est à effet immédiat.
            </para>
        </sect3>
    </sect2>

    <sect2 id="zend.db.table.row.serialize">
        <title>Sérialisation et désérialisation d'un Row</title>

        <para>
            Il peut être utile de sauvegarder le contenu d'un enregistrement (Row) sur un
            support quelconque, pour une utilisation ultérieure. La
            <emphasis>sérialisation</emphasis> est le nom de l'opération qui consiste à transformer
            un objet en une forme facilement stockable (dans un fichier par exemple). Les objets du
            type <classname>Zend_Db_Table_Row_Abstract</classname> sont sérialisables.
        </para>

        <sect3 id="zend.db.table.row.serialize.serializing">
            <title>Sérialiser un Row</title>

            <para>
                Utilisez simplement la fonction <acronym>PHP</acronym> <methodname>serialize()</methodname> pour créer une
                chaîne de caractères représentant votre objet Row.
            </para>

            <example id="zend.db.table.row.serialize.serializing.example">
                <title>Exemple de sérialisation d'un Row</title>

                <programlisting language="php"><![CDATA[
$bugs = new Bugs();
$row = $bugs->fetchRow('bug_id = 1');

// Convertit l'objet en une forme sérialisée
$serializedRow = serialize($row);

// Maintenant vous pouvez utiliser $serializedRow
// pour l'écrire dans un fichier, etc.
]]></programlisting>
            </example>
        </sect3>

        <sect3 id="zend.db.table.row.serialize.unserializing">
            <title>Désérialiser les données d'un Row</title>

            <para>
                Utilisez simplement la fonction <acronym>PHP</acronym> <methodname>unserialize()</methodname>. L'objet Row
                originel est alors recréé.
            </para>

            <para>
                Notez que l'objet retourné fonctionne alors en mode
                <emphasis>déconnecté</emphasis>. Vous pouvez lire les valeurs des colonnes, mais pas
                les modifier ni enregistrer l'objet en base de données
                (<methodname>save()</methodname>).
            </para>

            <example id="zend.db.table.row.serialize.unserializing.example">
                <title>Exemple de désérialisation d'un objet Row sérialisé</title>

                <programlisting language="php"><![CDATA[
$rowClone = unserialize($serializedRow);

// Vous ne pouvez faire qu'une utilisation en lecture seule
echo $rowClone->bug_description;
]]></programlisting>
            </example>

            <note>
                <title>Pourquoi ce mode déconnecté imposé ?</title>

                <para>
                    Un objet sérialisé est une chaîne de caractère, humainement visible. Il
                    est donc peu sécurisé d'y laisser un mot de passe vers un serveur de base de
                    données. Le lecteur d'un objet Row sérialisé ne devrait pas pouvoir accéder à la
                    base de données. De plus, une connexion à une base de données est un type non
                    sérialisable par <acronym>PHP</acronym> (ressource).
                </para>
            </note>
        </sect3>

        <sect3 id="zend.db.table.row.serialize.set-table">
            <title>Reconnecter l'objet Row à la Table</title>

            <para>
                Il est bien entendu possible de reconnecter l'objet Row à la base de données,
                et plus précisément à la Table dont il fut issu. Utilisez la méthode
                <methodname>setTable()</methodname> et passez lui une instance héritant de
                <classname>Zend_Db_Table_Abstract</classname>. Une fois reconnecté, l'objet Row
                possède de nouveau un accès à la base de données, et n'est donc plus en mode lecture
                seule.
            </para>

            <example id="zend.db.table.row.serialize.set-table.example">
                <title>Exemple de réactivation d'un Row</title>

                <programlisting language="php"><![CDATA[
$rowClone = unserialize($serializedRow);

$bugs = new Bugs();

// Reconnecte le Row à la table et donc, à la base de données
$rowClone->setTable($bugs);

// Maintenant il est possible de l'utiliser en mode écriture
$rowClone->bug_status = 'FIXED';
$rowClone->save();
]]></programlisting>
            </example>
        </sect3>
    </sect2>

    <sect2 id="zend.db.table.row.extending">
        <title>Étendre la classe Row</title>

        <para>
            Vous pouvez utilisez votre propre classe étendant
            <classname>Zend_Db_Table_Row_Abstract</classname>. Spécifiez votre classe dans la
            propriété protégée <varname>$_rowClass</varname> de la classe de votre Table, ou dans le
            tableau du constructeur de l'objet Table.
        </para>

        <example id="zend.db.table.row.extending.example">
            <title>Spécification d'une classe Row personnalisée</title>

            <programlisting language="php"><![CDATA[
class MyRow extends Zend_Db_Table_Row_Abstract
{
    // ...personnalisations
}

// Spécifie la classe de Row utilisée pour toutes les
// instance de la classe de Table
class Products extends Zend_Db_Table_Abstract
{
    protected $_name = 'products';
    protected $_rowClass = 'MyRow';
}

// Ou pour une classe de table spécifique, via son constructeur
$bugs = new Bugs(array('rowClass' => 'MyRow'));
]]></programlisting>
        </example>

        <sect3 id="zend.db.table.row.extending.overriding">
            <title>Initialisation et pré-traitements d'un Row</title>

            <para>
                Si vous avez un besoin spécifique d'implémenter une logique spéciale après la
                création d'une instance de Row, vous pouvez utiliser sa méthode <methodname>init()</methodname>,
                qui est appelée dans son constructeur, mais après que les méta données aient été
                calculées. <example id="zend.db.table.row.init.usage.example">
                        <title>Exemple d'utilisation de la méthode init()</title>

                        <programlisting language="php"><![CDATA[
class MyApplicationRow extends Zend_Db_Table_Row_Abstract
{
    protected $_role;

    public function init()
    {
        $this->_role = new MyRoleClass();
    }
}
]]></programlisting>
                    </example>
                </para>
        </sect3>

        <sect3 id="zend.db.table.row.extending.insert-update">
            <title>
                Définir sa propre logique pour Insert, Update, et Delete dans Zend_Db_Table_Row
            </title>

            <para>
                La classe des Rows appelle les méthodes protégées <methodname>_insert()</methodname>,
                <methodname>_update()</methodname>, et <methodname>_delete()</methodname> avant d'effectuer chacune des
                opérations respectives <constant>INSERT</constant>, <constant>UPDATE</constant>, et
                <constant>DELETE</constant>. Il est donc possible de définir sa propre logique dans votre
                sous-classe de Row.
            </para>

            <para>
                Ci-dessous vous trouverez des exemples d'utilisation d'une logique
                personnalisée dans les classes de Row :
            </para>

            <example id="zend.db.table.row.extending.overriding-example1">
                <title>Exemple de logique personnalisée dans une classe de Row</title>

                <para>
                    La logique personnelle peut donc être déportée dans une classe de Row qui
                    ne s'appliquera qu'à certaines tables, et pas à d'autres. Sinon, la classe de
                    Table utilise le Row par défaut.
                </para>

                <para>
                    Par exemple, vous souhaitez historiser toutes les insertions sur une Table
                    spécifique, mais uniquement si la configuration du site le permet :
                </para>

                <programlisting language="php"><![CDATA[
class MyLoggingRow extends Zend_Db_Table_Row_Abstract
{
    protected function _insert()
    {
        $log = Zend_Registry::get('database_log');
        $log->info(Zend_Debug::dump($this->_data,
                                    "INSERT: $this->_tableClass",
                                    false));
    }
}

// $loggingEnabled est une variable d'exemple qui définit si
// l'historisation est activée ou pas
if ($loggingEnabled) {
    $bugs = new Bugs(array('rowClass' => 'MyLoggingRow'));
} else {
    $bugs = new Bugs();
}
]]></programlisting>
            </example>

            <example id="zend.db.table.row.extending.overriding-example2">
                <title>Exemple d'une classe de Row qui historise les insertions de plusieurs
                tables</title>

                <para>
                    En passant l'objet Row personnalisé à chacune des Tables concernées, alors
                    vous n'aurez pas besoin de définir cette logique dans chacune des classes des
                    Tables.
                </para>

                <para>
                    Dans cet exemple, le code qui effectue l'historisation est identique à
                    celui de l'exemple précédent.
                </para>

                <programlisting language="php"><![CDATA[
class MyLoggingRow extends Zend_Db_Table_Row_Abstract
{
    protected function _insert()
    {
        $log = Zend_Registry::get('database_log');
        $log->info(Zend_Debug::dump($this->_data,
                                    "INSERT: $this->_tableClass",
                                    false));
    }
}

class Bugs extends Zend_Db_Table_Abstract
{
    protected $_name = 'bugs';
    protected $_rowClass = 'MyLoggingRow';
}

class Products extends Zend_Db_Table_Abstract
{
    protected $_name = 'products';
    protected $_rowClass = 'MyLoggingRow';
}
]]></programlisting>
            </example>
        </sect3>

        <sect3 id="zend.db.table.row.extending.inflection">
            <title>Définir l'inflexion dans Zend_Db_Table_Row</title>

            <para>
                Il peut être intéressant de personnaliser l'accès aux colonnes de la table
                représentée par un résultat Row, plutôt que d'utiliser le nom des colonnes telles
                que définies dans le SGBDR sous-jacent. La transformation de l'un vers l'autre est
                appelée <emphasis>inflexion</emphasis>.
            </para>

            <para>
                Les classes Zend_Db n'utilisent pas l'inflexion par défaut. Voyez <xref
                linkend="zend.db.table.extending.inflection" /> pour plus de détails sur ce
                procédé.
            </para>

            <para>
                Ainsi si vous voulez utiliser l'inflexion, vous devez implémenter vous-même la
                transformation à effectuer en redéfinissant la méthode
                <methodname>_transformColumn()</methodname> dans votre classe de Row, et bien entendu utiliser
                cette classe de Row pour votre Table.
            </para>

            <example id="zend.db.table.row.extending.inflection.example">
                <title>Exemple d'utilisation de l'inflexion</title>

                <para>
                    Ceci vous permet d'utiliser les accesseurs de votre Row de manière
                    transformée. La classe de votre Row utilisera <methodname>_transformColumn()</methodname>
                    pour changer le nom de la colonne appelée, avant de le faire correspondre à un
                    nom dans la table réelle de la base de données.
                </para>

                <programlisting language="php"><![CDATA[
class MyInflectedRow extends Zend_Db_Table_Row_Abstract
{
    protected function _transformColumn($columnName)
    {
        $nativeColumnName = myCustomInflector($columnName);
        return $nativeColumnName;
    }
}

class Bugs extends Zend_Db_Table_Abstract
{
    protected $_name = 'bugs';
    protected $_rowClass = 'MyInflectedRow';
}

$bugs = new Bugs();
$row = $bugs->fetchNew();

// Utilisez des nom de colonnes CamelCase, l'inflecteur les
// transformera alors pour vous afin d'établir la correspondance
// avec les noms natifs des colonnes.
$row->bugDescription = 'New description';
]]></programlisting>
            </example>

            <para>En revanche, c'est à vous d'écrire votre mécanisme d'inflexion.</para>
        </sect3>
    </sect2>
</sect1>