added experimental expand me...
|
1 |
=encoding utf8 |
2 | ||
updated document
|
3 |
=head1 NAME |
added experimental expand me...
|
4 | |
update pod
|
5 |
DBIx::Custom::Guide::Ja - DBIx::Customのガイドブック |
added experimental expand me...
|
6 | |
7 |
=head1 ガイド |
|
8 | ||
update pod
|
9 |
L<DBIx::Custom>はSQLの実行を簡単に行うためのクラスです。 |
10 |
L<DBIx::Class>やL<DBIx::Simple>と同じように |
|
updated document
|
11 |
L<DBI>のラッパクラスになっています。L<DBIx::Class>よりも簡単に、 |
12 |
L<DBIx::Simple>よりもはるかに柔軟なことを行うことができます。 |
|
added experimental expand me...
|
13 | |
14 |
L<DBIx::Custom>はO/Rマッパーではありません。O/Rマッパーは |
|
15 |
便利ですが、O/Rマッパのたくさんの文法を覚える必要があります。 |
|
updated document
|
16 |
また、O/Rマッパによって生成されたSQLは非効率なことがありますし、 |
17 |
複雑なSQLを生成することができないので、 |
|
update pod
|
18 |
生のSQLを実行しなければならない場合がたくさんあります。 |
added experimental expand me...
|
19 | |
updated document
|
20 |
L<DBIx::Custom>はO/Rマッパとは対照的な設計が行われています。 |
updated document
|
21 |
L<DBIx::Custom>の主な目的は、SQLを尊重しつつ、L<DBI>だけでは |
updated document
|
22 |
とてもめんどうな作業を簡単にすることです。もしSQLについて |
23 |
多くの知識を持っているならば、L<DBIx::Custom>でそのまま |
|
updated document
|
24 |
活用することができます。 |
updated document
|
25 | |
update pod
|
26 |
L<DBIx::Custom>の仕組みを少しだけ説明しておきます。 |
updated document
|
27 |
L<DBIx::Custom>では、タグと呼ばれるものを |
28 |
SQLの中に埋め込むことができます。 |
|
29 | ||
30 |
select * from book where {= title} and {=author}; |
|
31 | ||
32 |
{}で囲まれた部分がタグです。このSQLは実際に実行されるときには |
|
33 |
次のようにプレースホルダに展開されます。 |
|
34 | ||
updated document
|
35 |
select * from book where title = ? and author = ?; |
updated document
|
36 | |
37 |
これらの展開にはどのような意味があるのでしょうかと質問 |
|
update pod
|
38 |
されるかもしれません。この簡単な仕組みの上に |
39 |
便利な機能が実装されます。それは以下のようなものです。 |
|
updated document
|
40 | |
updated document
|
41 |
=over 4 |
updated document
|
42 | |
update pod
|
43 |
=item 1. プレースホルダにバインドする値をハッシュリファレンスで指定 |
updated document
|
44 | |
update pod
|
45 |
L<DBI>を使うのであればプレースホルダにバインドする値は配列 |
updated document
|
46 |
で指定する必要があります。 |
updated document
|
47 | |
updated document
|
48 |
$sth->execute(@bind); |
49 | ||
50 |
L<DBIx::Custom>を利用するのであればハッシュリファレンスで指定すること |
|
51 |
できます。 |
|
52 |
|
|
53 |
my $param = {title => 'Perl', author => 'Ken'}; |
|
54 |
$dbi->execute($sql, $param); |
|
55 | ||
update pod
|
56 |
=item 2. 値のフィルタリング |
updated document
|
57 | |
update pod
|
58 |
L<DBIx::Custom>はフィルタリングの機能を提供します。 |
updated document
|
59 |
たとえば、日付の列は、Perlで扱うときにはC<Time::Piece>などの日付オブジェクト |
60 |
で扱い、データベースに格納するときはデータベースの日付型に変換したい |
|
61 |
と思うのではないでしょうか。またデータベースから取り出すときは |
|
62 |
データベースの日付型から日付オブジェクトに変換したと思うのでは |
|
63 |
ないでしょうか。 |
|
64 | ||
65 |
このようなときはフィルタ機能を使うことができます。 |
|
66 | ||
67 |
まずフィルタを登録します。 |
|
68 | ||
69 |
$dbi->register_filter( |
|
70 |
tp_to_date => sub { |
|
71 |
... |
|
72 |
}, |
|
73 |
date_to_tp => sub { |
|
74 |
... |
|
75 |
} |
|
76 |
); |
|
77 | ||
78 |
次にテーブルの各列にこのフィルタを適用します。 |
|
79 | ||
80 |
$dbi->apply_filter('book', |
|
renamed dbi_options to dbi_o...
|
81 |
'issue_date' => {out => 'tp_to_date', in => 'date_to_tp'} |
updated document
|
82 |
); |
83 | ||
84 |
outはPerlからデータベースに保存する方向、inはデータベースからPerlに取得する方向です。 |
|
85 | ||
update pod
|
86 |
多くのメソッドで自動的にこのフィルタが有効になります。 |
updated document
|
87 | |
update pod
|
88 |
$dbi->insert(table => 'book', param => {issue_date => $tp}); |
updated document
|
89 | |
90 |
=item 3. 選択的な検索条件 |
|
91 | ||
update pod
|
92 |
L<DBI>では選択的に検索条件を作成することは難しいです。 |
updated document
|
93 | |
94 |
たとえば、検索条件にtitleとauthorが指定された場合は次のSQLを |
|
95 | ||
96 |
select * from book where title = ? and author = ?; |
|
97 | ||
98 |
titleだけの場合は次のSQLを |
|
99 | ||
100 |
select * from book where title = ?; |
|
101 |
|
|
update pod
|
102 |
authorだけの場合は次のSQLを実行した場合を考えましょう。 |
updated document
|
103 | |
104 |
select * from book where author = ?; |
|
105 | ||
106 |
これはとても大変な作業なので、通常はL<SQL::Abstract>を動的に生成してくれる |
|
107 |
モジュールを利用することになります。 |
|
108 | ||
109 |
L<DBIx::Custom>はさらに簡単で便利な方法を用意しています。 |
|
110 | ||
update pod
|
111 |
# Whereオブジェクト |
updated document
|
112 |
my $where = $dbi->where; |
update pod
|
113 |
|
114 |
# 検索条件 |
|
updated document
|
115 |
$where->clause( |
116 |
['and', '{= title}', {'= author'}] |
|
117 |
); |
|
update pod
|
118 |
|
119 |
# 必要な列を自動的に選択するための設定 |
|
120 |
$where->param({title => 'Perl'}); |
|
updated document
|
121 | |
update pod
|
122 |
# SQLへのWhere句の埋め込み |
updated document
|
123 |
my $sql = "select * from book $where"; |
124 | ||
125 |
詳しい説明は後ほど行いますが、上記のように記述すれば、 |
|
126 |
L<DBIx::Custom>では選択的な検索条件を持つWhere句を生成することができます。 |
|
127 |
検索条件が入れ子になった構造やorについても対応しています。 |
|
128 | ||
129 |
=item 4. 挿入、更新、削除、選択を行うためのメソッド |
|
130 | ||
update pod
|
131 |
L<DBIx::Custom>では挿入、更新、削除、選択を行うための |
132 |
メソッドを提供しています。 |
|
133 |
C<insert()>, C<update()>, C<delete()>,C<select()>などがあります。 |
|
updated document
|
134 | |
135 |
my $param = {title => 'Perl', author => 'Ken'}; |
|
136 |
$dbi->insert(table => 'book', param => $param); |
|
137 | ||
update pod
|
138 |
=item 5. テーブルのためのメソッドの登録 |
updated document
|
139 | |
update pod
|
140 |
テーブルのためにメソッドを登録することができます。 |
updated document
|
141 | |
update pod
|
142 |
$dbi->table('book')->method( |
updated document
|
143 |
list => sub { |
144 |
... |
|
145 |
}, |
|
update pod
|
146 |
something => sub { |
147 |
... |
|
updated document
|
148 |
} |
149 |
); |
|
150 | ||
update pod
|
151 |
メソッドの利用です。 |
updated document
|
152 | |
153 |
$dbi->table('book')->list; |
|
added experimental expand me...
|
154 | |
update pod
|
155 |
多くのO/Rマッパではテーブルのためのクラスを作成する必要がありますが、 |
156 |
L<DBIx::Custom>では簡単です。 |
|
added experimental expand me...
|
157 | |
updated document
|
158 |
=back |
159 | ||
update pod
|
160 |
L<DBIx::Custom>はとても便利です。 |
161 |
興味をもたれた方は、続きをご覧になってみてください。 |
|
added experimental expand me...
|
162 | |
163 |
=head2 1. データベースへの接続 |
|
164 | ||
update pod
|
165 |
L<DBIx::Custom>を読み込みます。 |
updated document
|
166 | |
167 |
use DBIx::Custom; |
|
168 | ||
update pod
|
169 |
データベースに接続するにはC<connect()>メソッドを使用します。 |
170 |
戻り値はL<DBIx::Custom>オブジェクトです。 |
|
added experimental expand me...
|
171 | |
updated document
|
172 |
my $dbi = DBIx::Custom->connect( |
update pod
|
173 |
data_source => "dbi:mysql:database=bookstore", |
updated document
|
174 |
user => 'ken', |
175 |
password => '!LFKD%$&', |
|
176 |
dbi_options => {mysql_enable_utf8 => 1} |
|
177 |
); |
|
added experimental expand me...
|
178 | |
update pod
|
179 |
C<data_source>はデータベースシステムに応じたものである必要があります。 |
180 |
以下はデータソースのサンプルです。 |
|
added experimental expand me...
|
181 | |
updated document
|
182 |
B<MySQL> |
added experimental expand me...
|
183 | |
deprecated DBIx::Custom::MyS...
|
184 |
"dbi:mysql:database=$database" |
185 |
"dbi:mysql:database=$database;host=$hostname;port=$port" |
|
added experimental expand me...
|
186 | |
updated document
|
187 |
B<SQLite> |
deprecated DBIx::Custom::MyS...
|
188 | |
189 |
"dbi:SQLite:dbname=$database" |
|
190 |
"dbi:SQLite:dbname=:memory:" |
|
191 | ||
updated document
|
192 |
B<PostgreSQL> |
deprecated DBIx::Custom::MyS...
|
193 | |
194 |
"dbi:Pg:dbname=$dbname" |
|
195 | ||
updated document
|
196 |
B<Oracle> |
deprecated DBIx::Custom::MyS...
|
197 | |
198 |
"dbi:Oracle:$dbname" |
|
199 |
"dbi:Oracle:host=$host;sid=$sid" |
|
200 | ||
updated document
|
201 |
B<ODBC(Microsoft Access)> |
deprecated DBIx::Custom::MyS...
|
202 | |
203 |
"dbi:ODBC:driver=Microsoft Access Driver (*.mdb);dbq=hoge.mdb" |
|
204 | ||
updated document
|
205 |
B<ODBC(SQL Server)> |
deprecated DBIx::Custom::MyS...
|
206 | |
207 |
"dbi:ODBC:driver={SQL Server};Server=(local);database=test;Trusted_Connection=yes;AutoTranslate=No;" |
|
added experimental expand me...
|
208 | |
update pod
|
209 |
認証が必要な場合は、C<user>とC<password>を指定できます。 |
updated document
|
210 | |
update pod
|
211 |
L<DBIx::Custom>はL<DBI>のラッパークラスです。 |
212 |
L<DBI>のデータベースハンドルは取得するにあhC<dbh()>を使用します。 |
|
added experimental expand me...
|
213 | |
214 |
my $dbh = $dbi->dbh; |
|
215 | ||
updated document
|
216 |
L<DBIx::Custom>ではデータベースハンドル属性にはデフォルトで次のものが設定されます。 |
added experimental expand me...
|
217 |
|
218 |
$dbi->dbh->{RaiseError} = 1; |
|
219 |
$dbi->dbh->{PrintError} = 0; |
|
220 |
$dbi->dbh->{AutoCommit} = 1; |
|
221 | ||
update pod
|
222 |
致命的なエラーが起こるとプログラムは終了します。 |
223 |
SQLが実行されると自動的にコミットされます。 |
|
added experimental expand me...
|
224 | |
updated document
|
225 |
=head2 2. 挿入、更新、削除、選択のためのメソッド |
added experimental expand me...
|
226 | |
update pod
|
227 |
下記のメソッドがあります。 |
added experimental expand me...
|
228 | |
update pod
|
229 |
=head3 行の挿入 C<insert()> |
added experimental expand me...
|
230 | |
update pod
|
231 |
データベースに行を挿入するにはC<insert()>を使用します。 |
added experimental expand me...
|
232 | |
remove DBIx::Custom::Model
|
233 |
$dbi->insert(table => 'book', |
added experimental expand me...
|
234 |
param => {title => 'Perl', author => 'Ken'}); |
235 | ||
update pod
|
236 |
C<table>はテーブル名、C<param>は挿入する行のデータです。 |
added experimental expand me...
|
237 | |
update pod
|
238 |
次のSQLが実行されます。 |
updated document
|
239 | |
240 |
insert into (title, author) values (?, ?); |
|
added experimental expand me...
|
241 | |
update pod
|
242 |
=head3 データの更新 C<update()> |
added experimental expand me...
|
243 | |
update pod
|
244 |
データベースの行を更新するには、C<update()>を使用します。 |
added experimental expand me...
|
245 | |
remove DBIx::Custom::Model
|
246 |
$dbi->update(table => 'book', |
added experimental expand me...
|
247 |
param => {title => 'Perl', author => 'Ken'}, |
248 |
where => {id => 5}); |
|
249 | ||
update pod
|
250 |
C<table>はテーブル名、C<param>は更新データ、C<where>は |
251 |
条件です。 |
|
added experimental expand me...
|
252 | |
update pod
|
253 |
次のSQLが実行されます。 |
updated document
|
254 | |
255 |
update book set title = ?, author = ?; |
|
added experimental expand me...
|
256 | |
update pod
|
257 |
安全のためC<where>のないupdate()を実効することはできません。 |
added experimental expand me...
|
258 |
もしすべての行を更新したい場合は |
updated document
|
259 |
C<update_all()>を使用してください。 |
added experimental expand me...
|
260 | |
remove DBIx::Custom::Model
|
261 |
$dbi->update_all(table => 'book', |
added experimental expand me...
|
262 |
param => {title => 'Perl', author => 'Ken'}); |
263 | ||
update pod
|
264 |
=head3 データの削除 C<delete()> |
added experimental expand me...
|
265 | |
update pod
|
266 |
データベースの行を1件削除するには、C<delete()>を使用します。 |
added experimental expand me...
|
267 | |
remove DBIx::Custom::Model
|
268 |
$dbi->delete(table => 'book', |
added experimental expand me...
|
269 |
where => {author => 'Ken'}); |
270 | ||
update pod
|
271 |
C<table>はテーブル名、C<where>は条件です。 |
added experimental expand me...
|
272 | |
update pod
|
273 |
次のSQLが実行されます。 |
updated document
|
274 | |
275 |
delete from book where id = ?; |
|
added experimental expand me...
|
276 | |
update pod
|
277 |
安全のためC<where>のないC<delete()>を実効することはできません。 |
added experimental expand me...
|
278 |
もしすべての行を削除したい場合は |
updated document
|
279 |
C<delete_all()>を使用してください。 |
added experimental expand me...
|
280 | |
remove DBIx::Custom::Model
|
281 |
$dbi->delete_all(table => 'book'); |
added experimental expand me...
|
282 | |
update pod
|
283 |
=head3 データの選択 C<select()> |
added experimental expand me...
|
284 | |
updated document
|
285 |
行を選択するにはC<select()>を使用します。 |
added experimental expand me...
|
286 | |
remove DBIx::Custom::Model
|
287 |
my $result = $dbi->select(table => 'book'); |
added experimental expand me...
|
288 | |
update pod
|
289 |
次のSQLが実行されます。 |
added experimental expand me...
|
290 | |
updated document
|
291 |
select * from book; |
added experimental expand me...
|
292 | |
update pod
|
293 |
戻り値はL<DBIx::Custom::Result> |
updated document
|
294 |
オブジェクトです。行をフェッチするにはC<fetch()>を使用します。 |
added experimental expand me...
|
295 | |
296 |
while (my $row = $result->fetch) { |
|
297 |
my $title = $row->[0]; |
|
298 |
my $author = $row->[1]; |
|
299 |
} |
|
300 | ||
update pod
|
301 |
L<DBIx::Custom::Result>についてはL<3. 行のフェッチ/"3. 行のフェッチ">を見てください。 |
updated document
|
302 | |
update pod
|
303 |
サンプルを続けます。 |
added experimental expand me...
|
304 | |
305 |
my $result = $dbi->select( |
|
remove DBIx::Custom::Model
|
306 |
table => 'book', |
updated document
|
307 |
column => ['author', 'title'], |
added experimental expand me...
|
308 |
where => {author => 'Ken'} |
309 |
); |
|
310 | ||
update pod
|
311 |
C<column>は列名、C<where>は条件です。 |
312 | ||
313 |
次のSQLが実行されます。 |
|
updated document
|
314 | |
315 |
select author, title from book where author = ?; |
|
added experimental expand me...
|
316 | |
update pod
|
317 |
次のサンプルです。 |
added experimental expand me...
|
318 | |
319 |
my $result = $dbi->select( |
|
remove DBIx::Custom::Model
|
320 |
table => ['book', 'rental'], |
updated document
|
321 |
where => {book.name => 'Perl'}, |
remove DBIx::Custom::Model
|
322 |
relation => {'book.id' => 'rental.book_id'} |
added experimental expand me...
|
323 |
); |
324 | ||
update pod
|
325 |
C<relation>テーブル間の関係です。これは内部結合です。 |
326 | ||
327 |
次のSQLが実行されます。 |
|
328 | ||
updated document
|
329 |
bookテーブルのid列とrentalテーブルのbook_idが関連付けられます。 |
update pod
|
330 |
次のSQLが実行されます。 |
added experimental expand me...
|
331 | |
updated document
|
332 |
select * from book, rental where book.name = ? and book.id = rental.book_id; |
added experimental expand me...
|
333 | |
update pod
|
334 |
次のサンプルです。 |
added experimental expand me...
|
335 | |
336 |
my $result = $dbi->select( |
|
remove DBIx::Custom::Model
|
337 |
table => 'book', |
added experimental expand me...
|
338 |
where => {author => 'Ken'}, |
updated document
|
339 |
append => 'for update', |
added experimental expand me...
|
340 |
); |
341 | ||
update pod
|
342 |
C<append>はSQLの末尾に追加される文字列です。 |
343 | ||
344 |
次のSQLが実行されます。 |
|
updated document
|
345 | |
updated document
|
346 |
select * book where author = ? for update; |
updated document
|
347 | |
updated document
|
348 |
またC<append>は、C<select>だけでなくC<insert()>、C<update()>、C<update_all()> |
349 |
C<delete()>、C<delete_all()>、C<select()>で使用することもできます。 |
|
updated document
|
350 | |
update pod
|
351 |
=head3 SQLの実行 C<execute()> |
removed experimental txn_sco...
|
352 | |
update pod
|
353 |
SQLを実行するにはC<execute()>を使用します。 |
removed experimental txn_sco...
|
354 | |
355 |
$dbi->execute("select * from book;"); |
|
356 | ||
update pod
|
357 |
タグを処理してSQLを実行します。 |
removed experimental txn_sco...
|
358 | |
359 |
$dbi->execute( |
|
360 |
"select * from book {= title} and {= author};" |
|
361 |
param => {title => 'Perl', author => 'Ken'} |
|
362 |
); |
|
363 | ||
update pod
|
364 |
次のSQLが実行されます。 |
removed experimental txn_sco...
|
365 | |
366 |
select * from book title = ? and author = ?; |
|
367 | ||
update pod
|
368 |
プレースホルダにtitleとauthorの値が埋め込まれます。 |
removed experimental txn_sco...
|
369 | |
update pod
|
370 |
タグについてはL<5. タグ/"5. タグ">を見てください。 |
removed experimental txn_sco...
|
371 | |
update pod
|
372 |
またC<execute()>のSQLの末尾にはセミコロンを置く必要はありません。 |
removed experimental txn_sco...
|
373 | |
374 |
$dbi->execute('select * from book'); |
|
375 | ||
updated document
|
376 |
=head2 3. 行のフェッチ |
377 | ||
updated document
|
378 |
C<select()>メソッドの戻り値はL<DBIx::Custom::Result>オブジェクトです。 |
update pod
|
379 |
行をフェッチするためのさまざまなメソッドがあります。 |
updated document
|
380 | |
update pod
|
381 |
=head3 1行づつフェッチ(配列) C<fetch()> |
updated document
|
382 | |
updated document
|
383 |
一行フェッチして配列のリファレンスに格納するにはC<fetch()>を使用します。 |
updated document
|
384 | |
update pod
|
385 |
my $row = $result->fetch; |
386 | ||
387 |
以下のようにすべての行を取得することができます。 |
|
388 | ||
updated document
|
389 |
while (my $row = $result->fetch) { |
updated document
|
390 |
my $title = $row->[0]; |
391 |
my $author = $row->[1]; |
|
updated document
|
392 |
} |
393 | ||
update pod
|
394 |
=head3 最初の行だけフェッチ(配列) C<fetch_first()> |
updated document
|
395 | |
updated document
|
396 |
一行だけフェッチして配列のリファレンスに格納するにはC<fetch_first()> |
397 |
を使用します。 |
|
updated document
|
398 | |
399 |
my $row = $result->fetch_first; |
|
400 | ||
update pod
|
401 |
ステートメントハンドルのC<finish()>が実行される |
402 |
ので残りの行をフェッチできません。 |
|
updated document
|
403 | |
update pod
|
404 |
=head3 複数行を順にフェッチ(配列) C<fetch_multi()> |
updated document
|
405 | |
updated document
|
406 |
複数行をフェッチして配列のリファレンスを要素に持つ |
407 |
配列のリファレンスに格納するにはC<fetch_multi()>を使用します。 |
|
updated document
|
408 | |
updated document
|
409 |
while (my $rows = $result->fetch_multi(2)) { |
410 |
my $title0 = $rows->[0][0]; |
|
411 |
my $author0 = $rows->[0][1]; |
|
412 |
|
|
413 |
my $title1 = $rows->[1][0]; |
|
414 |
my $author1 = $rows->[1][1]; |
|
updated document
|
415 |
} |
416 | ||
update pod
|
417 |
取り出したい行数を引数に指定します。 |
updated document
|
418 | |
update pod
|
419 |
次のようなデータを取得できます。 |
updated document
|
420 | |
421 |
[ |
|
422 |
['Perl', 'Ken'], |
|
423 |
['Ruby', 'Mark'] |
|
424 |
] |
|
425 | ||
update pod
|
426 |
=head3 すべての行をフェッチ(配列) C<fetch_all> |
updated document
|
427 | |
updated document
|
428 |
すべての行をフェッチして配列のリファレンスを要素に持つ |
429 |
配列のリファレンスに格納するにはC<fetch_all()>を使用します。 |
|
updated document
|
430 | |
431 |
my $rows = $result->fetch_all; |
|
432 | ||
updated document
|
433 |
すべての行を格納した次のようなデータを取得できます。 |
434 | ||
435 |
[ |
|
436 |
['Perl', 'Ken'], |
|
437 |
['Ruby', 'Mark'] |
|
438 |
] |
|
439 | ||
update pod
|
440 |
=head3 1行づつフェッチ(ハッシュ) C<fetch_hash()> |
updated document
|
441 | |
updated document
|
442 |
一行フェッチしてハッシュのリファレンスに格納するにはC<fetch_hash()>を使用します。 |
updated document
|
443 | |
444 |
while (my $row = $result->fetch_hash) { |
|
445 |
my $title = $row->{title}; |
|
446 |
my $author = $row->{author}; |
|
447 |
} |
|
448 | ||
update pod
|
449 |
=head3 最初の行だけフェッチ(ハッシュ) C<fetch_hash_first()> |
updated document
|
450 | |
updated document
|
451 |
一行だけフェッチしてハッシュのリファレンスに格納するには |
452 |
C<fetch_hash_first()>を使用します。 |
|
updated document
|
453 | |
454 |
my $row = $result->fetch_hash_first; |
|
455 | ||
update pod
|
456 |
ステートメントハンドルのC<finish()>が実行される |
457 |
ので残りの行をフェッチできません。 |
|
updated document
|
458 | |
update pod
|
459 |
=head3 複数行をフェッチ(ハッシュ) C<fetch_hash_multi()> |
updated document
|
460 | |
updated document
|
461 |
複数行をフェッチしてハッシュのリファレンスを要素に持つ |
462 |
配列のリファレンスに格納するにはC<fetch_hash_multi()> |
|
463 |
を使用します。 |
|
updated document
|
464 | |
465 |
while (my $rows = $result->fetch_hash_multi(5)) { |
|
updated document
|
466 |
my $title0 = $rows->[0]{title}; |
467 |
my $author0 = $rows->[0]{author}; |
|
468 |
my $title1 = $rows->[1]{title}; |
|
469 |
my $author1 = $rows->[1]{author}; |
|
updated document
|
470 |
} |
471 | ||
updated document
|
472 |
引数には取り出したい行数を指定します。 |
473 | ||
update pod
|
474 |
次のようなデータを取得できます。 |
updated document
|
475 | |
476 |
[ |
|
477 |
{title => 'Perl', author => 'Ken'}, |
|
478 |
{title => 'Ruby', author => 'Mark'} |
|
479 |
] |
|
480 | ||
update pod
|
481 |
=head3 すべての行をフェッチ(ハッシュ) C<fetch_hash_all()> |
updated document
|
482 | |
updated document
|
483 |
すべての行をフェッチしてハッシュのリファレンスを要素に持つ |
484 |
配列のリファレンスに格納するにはC<fetch_hash_all()> |
|
485 |
を使用します。 |
|
updated document
|
486 | |
487 |
my $rows = $result->fetch_hash_all; |
|
488 | ||
update pod
|
489 |
次のようなデータを取得できます。 |
updated document
|
490 | |
491 |
[ |
|
492 |
{title => 'Perl', author => 'Ken'}, |
|
493 |
{title => 'Ruby', author => 'Mark'} |
|
494 |
] |
|
495 | ||
update pod
|
496 |
=head3 ステートメントハンドル C<sth()> |
updated document
|
497 | |
update pod
|
498 |
ステートメントハンドル取得したい場合は |
499 |
<sth()>を使用します。 |
|
updated document
|
500 | |
501 |
my $sth = $result->sth; |
|
502 | ||
renamed dbi_options to dbi_o...
|
503 |
=head2 4. フィルタリング |
504 | ||
505 |
データベースにデータを登録するときやデータベースからデータを取得する |
|
506 |
ときに自動的に値の変換を行いたい場合が多いと思います。 |
|
507 |
たとえば、日付を表現する列の場合は、 |
|
508 |
データベースに登録する場合はL<Time::Piece>オブジェクトから |
|
509 |
データベースの日付のフォーマットに、 |
|
510 |
データベースからデータを取得するときは、その逆を行えると便利です。 |
|
511 | ||
update pod
|
512 |
=head3 フィルタの登録 C<register_filter()> |
renamed dbi_options to dbi_o...
|
513 | |
514 |
フィルタを登録するにはC<register_filter()>を使用します。 |
|
515 | ||
516 |
$dbi->register_filter( |
|
517 |
# Time::Piece object to DATE format |
|
518 |
tp_to_date => sub { |
|
519 |
my $date = shift; |
|
520 | ||
521 |
return '0000-00-00' unless $tp; |
|
522 |
return $tp->strftime('%Y-%m-%d'); |
|
523 |
}, |
|
524 |
|
|
525 |
# DATE to Time::Piece object |
|
526 |
date_to_tp => sub { |
|
527 |
my $date = shift; |
|
528 | ||
529 |
return if $date eq '0000-00-00'; |
|
530 |
return Time::Piece->strptime($date, '%Y-%m-%d'); |
|
531 |
}, |
|
532 |
); |
|
533 | ||
534 |
登録したフィルタはC<apply_filter()>などで利用することができます。 |
|
535 | ||
update pod
|
536 |
=head3 フィルタの適用 C<apply_filter()> |
renamed dbi_options to dbi_o...
|
537 | |
538 |
作成したフィルタを適用するには、C<apply_filter()>を使用します。 |
|
539 | ||
540 |
$dbi->apply_filter('book', |
|
541 |
issue_date => {out => 'tp_to_date', in => 'date_to_tp'}, |
|
542 |
first_issue_date => {out => 'tp_to_date', in => 'date_to_tp'} |
|
543 |
); |
|
544 | ||
545 |
第一引数はテーブル名です。第二引数以降は、列名とフィルタルールのペアを記述します。 |
|
546 |
フィルタルールのoutには、データベースにデータを送信するときに適用するフィルタを、 |
|
547 |
フィルタルールのinには、データベースからデータを取得するときに適用するフィルタを |
|
548 |
記述します。outがデータベースに送信する方向、inがデータベースから取り出す方向です。 |
|
549 |
フィルタには、C<register_filter>で登録したフィルタ名の他に、コードリファレンスを |
|
550 |
指定することもできます。 |
|
551 | ||
552 |
issue_date => {out => sub { ... }, in => sub { ... }} |
|
553 | ||
554 |
適用されたフィルタはC<insert()>、C<update()>、C<update_all()>、C<delete()>、 |
|
555 |
C<delete_all()>、C<select()>で有効になります。 |
|
556 | ||
557 |
my $tp = Time::Piece->strptime('2010/10/14', '%Y/%m/%d'); |
|
558 |
my $result = $dbi->select(table => 'book', where => {issu_date => $tp}); |
|
559 | ||
560 |
データベースにデータが送信されるときに、L<Time::Piece>オブジェクトは |
|
561 |
データベースの日付のフォーマット「2010-10-14」に変換されます。 |
|
562 | ||
563 |
また逆にデータをフェッチするときには、データベースの日付のフォーマットは |
|
564 |
タイムピースオブジェクトに変換されます。 |
|
565 | ||
566 |
my $row = $resutl->fetch_hash_first; |
|
567 |
my $tp = $row->{issue_date}; |
|
568 | ||
569 |
このような自動的に実行されるフィルタを登録できることがL<DBIx::Custom>の |
|
570 |
特徴のひとつです。 |
|
571 | ||
removed experimental expand
|
572 |
C<apply_filter()>で適用されたフィルタはテーブル名をを含む列名に対しても有効です。 |
573 | ||
574 |
$dbi->select( |
|
575 |
table => 'book', |
|
576 |
where => {'book.title' => 'Perl', 'book.author' => 'Ken'} |
|
577 |
); |
|
578 | ||
579 |
テーブルを区別する必要があるときに便利な機能です。 |
|
580 | ||
update pod
|
581 |
=head3 個別のフィルタの適用 C<filter> |
renamed dbi_options to dbi_o...
|
582 | |
583 |
C<apply_filter()>を使って最初にすべてのテーブルの列について |
|
removed experimental txn_sco...
|
584 |
フィルタを定義することもできますが、 |
585 |
個別にフィルタを適用することもできます。 |
|
renamed dbi_options to dbi_o...
|
586 |
個別のフィルタはC<apply_filter()>で適用したフィルタを上書きます。 |
removed experimental txn_sco...
|
587 |
個別のフィルタはSQLのasを使って、列の別名を作成する必要がある場合に活躍します。 |
588 | ||
589 |
データベースに送信する場合に、個別のフィルタを適用するには、各メソッドの |
|
590 |
C<filter>オプションを使用します。個別のフィルタは、C<insert()>、C<update()>、 |
|
591 |
C<update_all()>、C<delete()>、C<delete_all()>、C<select()>、C<execute()> |
|
592 |
で使用することができます。 |
|
renamed dbi_options to dbi_o...
|
593 | |
removed experimental txn_sco...
|
594 |
C<insert()>の例を示します。 |
595 | ||
596 |
$dbi->insert( |
|
597 |
table => 'book', |
|
598 |
param => {issue_date => $tp, first_issue_date => $tp}, |
|
599 |
filter => {issue_date => 'tp_to_date', first_issue_date => 'tp_to_date'} |
|
600 |
); |
|
601 | ||
602 |
C<execute()>の例を示します。 |
|
603 | ||
604 |
my $sql = <<"EOS"; |
|
605 |
select YEAR(issue_date) as issue_year |
|
606 |
from book |
|
607 |
where YEAR(issue_date) = {? issue_year} |
|
608 |
EOS |
|
609 |
|
|
610 |
my $result = $dbi->execute( |
|
611 |
$sql, |
|
612 |
param => {issue_year => '2010'}, |
|
613 |
filter => {issue_year => 'tp_to_year'} |
|
614 |
); |
|
615 | ||
616 |
これはC<filter>を使う良くある例です。issue_dateの変換についてはC<apply_filter()> |
|
617 |
で登録してあるのですが、新しく作成した列であるissue_yearについては、 |
|
618 |
何の変換も登録されていません。ですので、個別にフィルタを設定しています。 |
|
619 | ||
620 |
また反対に行をフェッチするときにも個別のフィルタを適用することができます。 |
|
621 |
フィルタを適用するには、 |
|
renamed dbi_options to dbi_o...
|
622 |
C<DBIx::Custom::Result>クラスのC<filter>メソッドを使用します。 |
623 | ||
removed experimental txn_sco...
|
624 |
$result->filter(issue_year => 'year_to_tp'); |
625 | ||
626 |
頻繁に利用するのであれば、個別に登録するよりもC<apply_filter()>で登録 |
|
627 |
しておいたほうが便利でしょう。C<apply_filter()>は存在しない列に対しても |
|
628 |
フィルタを適用できるからです。 |
|
629 | ||
630 |
$dbi->apply_filter('book', |
|
631 |
'issue_year' => {out => 'tp_to_year', in => 'year_to_tp'} |
|
632 |
); |
|
633 | ||
update pod
|
634 |
=head3 最終出力のためのフィルタリング C<end_filter()> |
635 | ||
removed experimental txn_sco...
|
636 |
C<DBIx::Custom::Result>ではさらに最後にもう一度、フィルタを追加で |
637 |
登録することができます。たとえばHTMLに出力したい場合に、Time::Piece |
|
638 |
オブジェクトから読みやすい記述に変換することができます。 |
|
639 |
最後のフィルタを登録するには、C<end_filter()>を使用します。 |
|
640 | ||
641 |
$result->end_filter(issue_date => sub { |
|
642 |
my $tp = shift; |
|
643 |
|
|
644 |
return '' unless $tp; |
|
645 |
return $tp->strftime('%Y/%m/%d %h:%m:%s (%a)'); |
|
646 |
}); |
|
647 | ||
648 |
日付を見やすい形にフォーマットすることができます。 |
|
649 | ||
650 |
フィルタはフェッチを行う前に登録しておく必要があることに |
|
651 |
注意してください。 |
|
652 | ||
653 |
$result->filter(...); |
|
654 |
$result->end_filter(...); |
|
655 |
my $row = $result->fetch_hash_first; |
|
656 | ||
update pod
|
657 |
=head3 列の情報を元にフィルタを適用する C<each_column()> |
removed experimental txn_sco...
|
658 | |
659 |
日付型の列は手動で設定しなくても、自動的に設定できると便利です。 |
|
660 |
このためにデータベースのテーブルの列のすべての情報を |
|
661 |
順番に処理するためのC<each_column()>があります。 |
|
662 | ||
663 |
$dbi->each_column( |
|
664 |
sub { |
|
665 |
my ($self, $table, $column, $info) = @_; |
|
666 |
|
|
667 |
my $type = $info->{TYPE_NAME}; |
|
668 |
|
|
669 |
my $filter = $type eq 'DATE' ? {out => 'tp_to_date', in => 'date_to_tp'} |
|
670 |
: $type eq 'DATETIME' ? {out => 'tp_to_datetime', in => 'datetime_to_tp'} |
|
671 |
: undef; |
|
672 |
|
|
673 |
$self->apply_filter($table, $column, $filter) |
|
674 |
if $filter; |
|
675 |
} |
|
676 |
); |
|
renamed dbi_options to dbi_o...
|
677 | |
removed experimental txn_sco...
|
678 |
each_columnはコールバックを受け取ります。コールバックの引数は |
679 |
順番にL<DBIx::Custom>オブジェクト、テーブル名、列名、列の情報です。 |
|
680 |
列の型名の情報をもとに自動的に、フィルタを適用しています。 |
|
681 | ||
682 |
ひとつの注意点としてコールバックの中から、コールバックの外側 |
|
683 |
の変数を参照しないように注意してください。each_columnは |
|
684 |
高々1回だけ実行されるだけなので、ほとんどの場合問題ありませんが、 |
|
685 |
循環参照によるメモリリークが発生してしまう可能性を持っているからです。 |
|
renamed dbi_options to dbi_o...
|
686 | |
687 |
=head2 5. タグ |
|
updated document
|
688 | |
update pod
|
689 |
=head3 タグの機能 |
updated document
|
690 | |
update pod
|
691 |
L<DBIx::Custom>はSQLの中にタグを埋め込む機能を持っています。 |
updated document
|
692 | |
update pod
|
693 |
select * from book where {= title} and {like author}; |
updated document
|
694 | |
update pod
|
695 |
{= title}と{like author}の部分がタグです。タグは次のような形式 |
696 |
を持ちます。 |
|
updated document
|
697 | |
update pod
|
698 |
{タグ名 引数1 引数2 ...} |
699 |
|
|
700 |
タグはC<{>で始まり、C<}>で終わります。最初のC<{>とタグ名の間 |
|
701 |
には空白を挿入しないよう注意してください。 |
|
updated document
|
702 | |
update pod
|
703 |
タグの機能のためにC<{>とC<}>は予約語になっています。 |
704 |
もし利用したい場合は直前に\をおいてエスケープを行う必要があります。 |
|
updated document
|
705 | |
update pod
|
706 |
select from book \\{ ... \\} |
707 | ||
708 |
C<\>自体がPerlのエスケープ文字ですので、 |
|
709 |
エスケープする場合はC<\>が二つ必要になるという点に注意してください。 |
|
710 | ||
711 |
上記のタグはSQLが実行される前に次のSQLに展開されます。 |
|
712 | ||
713 |
select * from book where title = ? and author like ?; |
|
updated document
|
714 | |
update pod
|
715 |
タグを含むSQLを実行するにはC<execute()>を使用します。 |
updated document
|
716 | |
update pod
|
717 |
my $sql = "select * from book where {= author} and {like title};" |
718 |
$dbi->execute($sql, param => {title => 'Perl', author => '%Ken%'}); |
|
updated document
|
719 | |
update pod
|
720 |
C<param>オプションを使って、プレースホルダに埋め込みたい値を |
721 |
ハッシュリファレンスで指定することができます。 |
|
updated document
|
722 | |
update pod
|
723 |
他のメソッドと同様にC<execute()>においてもC<filter>を指定することができます。 |
724 | ||
725 |
$dbi->execute($sql, param => {title => 'Perl', author => '%Ken%'} |
|
726 |
filter => {title => 'to_something'); |
|
727 | ||
728 |
C<execute>のひとつの注意点としてはC<apply_filter()>で適用されたフィルタ |
|
729 |
はデフォルトでは有効ではないということに注意してください。 |
|
730 |
C<apply_filter()>で適用されたフィルタを有効にするには、 |
|
731 |
C<table>を指定する必要があります。 |
|
732 | ||
733 |
$dbi->execute($sql, param => {title => 'Perl', author => '%Ken%'} |
|
734 |
table => ['book']); |
|
735 | ||
update pod
|
736 |
=head3 タグ一覧 |
update pod
|
737 | |
738 |
L<DBIx::Custom>では次のタグが使用可能です。 |
|
updated document
|
739 | |
740 |
このようにタグを使ってSQL文を表現するのがL<DBIx::Custom>の |
|
741 |
特徴です。以下のタグが利用可能です。 |
|
742 | ||
update pod
|
743 |
=head4 C<?> |
update pod
|
744 | |
745 |
C<?>タグは以下のように展開されます。 |
|
746 | ||
747 |
{? NAME} -> ? |
|
748 | ||
update pod
|
749 |
=head4 C<=> |
update pod
|
750 | |
751 |
C<=>タグは以下のように展開されます。 |
|
752 | ||
753 |
{= NAME} -> NAME = ? |
|
754 | ||
update pod
|
755 |
=head4 C<E<lt>E<gt>> |
update pod
|
756 | |
757 |
C<E<lt>E<gt>>タグは以下のように展開されます。 |
|
758 | ||
759 |
{<> NAME} -> NAME <> ? |
|
760 | ||
update pod
|
761 |
=head4 C<E<lt>> |
update pod
|
762 | |
763 |
C<E<lt>>タグは以下のように展開されます。 |
|
764 | ||
765 |
{< NAME} -> NAME < ? |
|
766 | ||
update pod
|
767 |
=head4 C<E<gt>> |
update pod
|
768 | |
769 |
C<E<gt>>タグは以下のように展開されます。 |
|
770 | ||
771 |
{> NAME} -> NAME > ? |
|
772 | ||
update pod
|
773 |
=head4 C<E<gt>=> |
update pod
|
774 | |
775 |
C<E<gt>=>タグは以下のように展開されます。 |
|
776 | ||
777 |
{>= NAME} -> NAME >= ? |
|
778 | ||
update pod
|
779 |
=head4 C<E<lt>=> |
update pod
|
780 | |
781 |
C<E<lt>=>タグは以下のように展開されます。 |
|
782 | ||
783 |
{<= NAME} -> NAME <= ? |
|
784 | ||
update pod
|
785 |
=head4 C<like> |
update pod
|
786 | |
787 |
C<like>タグは以下のように展開されます。 |
|
788 | ||
789 |
{like NAME} -> NAME like ? |
|
790 | ||
update pod
|
791 |
=head4 C<in> |
update pod
|
792 | |
793 |
C<in>タグは以下のように展開されます。プレースホルダの |
|
794 |
数を引数で指定する必要があることに注意してください。 |
|
795 | ||
796 |
{in NAME COUNT} -> NAME in [?, ?, ..] |
|
797 | ||
update pod
|
798 |
=head4 C<insert_param> |
update pod
|
799 | |
800 |
C<insert_param>タグは以下のように展開されます。 |
|
801 | ||
updated document
|
802 |
{insert_param NAME1 NAME2} -> (NAME1, NAME2) values (?, ?) |
update pod
|
803 | |
update pod
|
804 |
=head4 C<update_param> |
update pod
|
805 | |
806 |
C<update_param>タグは以下のように展開されます。 |
|
807 | ||
updated document
|
808 |
{update_param NAME1 NAME2} -> set NAME1 = ?, NAME2 = ? |
809 | ||
update pod
|
810 |
=head3 同名の列の扱い |
updated document
|
811 | |
update pod
|
812 |
同名の列を含むタグがある場合にも、SQLを実行することができます。 |
813 |
たとえば、二つの日付で比較しなければならない場合を |
|
814 |
考えて見ましょう。 |
|
updated document
|
815 | |
update pod
|
816 |
my $sql = "select * from table where {> date} and {< date};"; |
updated document
|
817 | |
update pod
|
818 |
このような場合は対応するパラメータの値を配列のリファレンスにします。 |
removed experimental txn_sco...
|
819 | |
update pod
|
820 |
my $dbi->execute($sql, param => {date => ['2010-10-01', '2012-02-10']}); |
821 | ||
update pod
|
822 |
=head3 タグの追加 C<register_tag()> |
update pod
|
823 | |
cleanup
|
824 |
L<DBIx::Custom>ではタグを独自に追加することができます。 |
update pod
|
825 |
タグを追加するにはC<register_tag()>を使用します。 |
removed experimental txn_sco...
|
826 | |
827 |
$dbi->register_tag( |
|
update pod
|
828 |
'=' => sub { |
829 |
my $column = shift; |
|
830 |
|
|
831 |
return ["$column = ?", [$column]]; |
|
removed experimental txn_sco...
|
832 |
} |
833 |
); |
|
834 | ||
update pod
|
835 |
ここではデフォルトのC<=>タグがどのように実装されているかを示しています。 |
836 |
タグを登録する関数の引数はタグの中に書かれた引数になります。 |
|
837 | ||
838 |
{タグ名 引数1 引数2} |
|
839 | ||
840 |
C<=>タグの場合は |
|
841 | ||
842 |
{= title} |
|
843 | ||
844 |
という形式ですから、サブルーチンにはtitleというひとつの列名がわたってきます。 |
|
845 | ||
846 |
サブルーチンの戻り値には次の形式の配列のリファレンスを返す必要があります。 |
|
847 | ||
848 |
[ |
|
849 |
展開後の文字列, |
|
850 |
[プレースホルダに埋め込みに利用する列名1, 列名2, ...] |
|
851 |
] |
|
852 | ||
853 |
一つ目の要素は展開後の文字列です。この例では |
|
854 | ||
855 |
'title = ?' |
|
856 | ||
857 |
を返す必要があります。 |
|
858 | ||
859 |
二つ目の要素はプレースホルダに埋め込みに利用する列名を含む配列の |
|
860 |
リファレンスです。今回の例では |
|
861 | ||
862 |
['title'] |
|
863 | ||
864 |
を返す必要があります。複数のプレースホルダを含む場合は、この部分が |
|
865 |
複数になります。C<insert_param>タグやC<update_param>タグは |
|
866 |
この部分が実際複数になっています。 |
|
867 | ||
868 |
上記を合わせると |
|
869 | ||
870 |
['title = ?', ['title']] |
|
871 |
|
|
872 |
を返す必要があるということです。 |
|
873 | ||
added experimental not_exist...
|
874 |
タグの実装の他のサンプルはL<DBIx::Custom::Tag>のソースコード |
875 |
をご覧になってみてください。 |
|
876 | ||
removed experimental txn_sco...
|
877 |
=head2 6. Where句の動的な生成 |
878 | ||
added experimental not_exist...
|
879 |
=head3 Where句の動的な生成 where() |
880 | ||
881 |
複数の検索条件を指定して、検索を行いたい場合があります。 |
|
882 |
次の3つのケースのwhere句を考えてみましょう。 |
|
883 |
下記のようなwhere句が必要になります。 |
|
884 | ||
885 |
titleの値だけで検索したい場合 |
|
886 | ||
887 |
where {= title} |
|
888 | ||
889 |
authorの値だけで検索したい場合 |
|
890 | ||
891 |
where {= author} |
|
892 | ||
893 |
titleとauthorの両方の値で検索したい場合 |
|
894 | ||
895 |
where {= title} and {=author} |
|
896 | ||
897 |
L<DBIx::Custom>では動的なWhere句の生成をサポートしています。 |
|
898 |
まずC<where()>でL<DBIx::Custom::Where>オブジェクトを生成します。 |
|
899 | ||
900 |
my $where = $dbi->where; |
|
901 | ||
902 |
次にC<clause()>を使用してwhere句を記述します。 |
|
903 | ||
904 |
$where->clause( |
|
905 |
['and', '{= title'}, '{= author}'] |
|
906 |
); |
|
907 | ||
908 |
clauseの指定方法は次のようになります。 |
|
909 | ||
910 |
['or' あるいは 'and', タグ1, タグ2, タグ3] |
|
911 | ||
912 |
第一引数にはorあるいはandを指定します。第二引数以降には |
|
913 |
検索条件をタグを使って記述します。 |
|
914 | ||
915 |
C<clause>の指定は入れ子にすることもでき、さらに複雑な条件 |
|
916 |
を記述することもできます。 |
|
917 | ||
918 |
['and', |
|
919 |
'{= title}', |
|
920 |
['or', '{= author}', '{like date}'] |
|
921 |
] |
|
922 | ||
923 |
このようにC<clause>を設定した後にC<param>にパラメータを指定します。 |
|
924 |
|
|
925 |
my $param => {title => 'Perl'}; |
|
926 |
$where->param($param); |
|
927 | ||
928 |
この例ではtitleだけがパラメータに含まれています。 |
|
929 | ||
930 |
この後C<to_string()>を実行すると$paramに含まれるパラメータを満たす |
|
931 |
where句を生成することができます。 |
|
932 | ||
933 |
my $where_clause = $where->to_string; |
|
934 | ||
935 |
パラメータはtitleだけですので、次のようなwhere句が生成されます。 |
|
936 | ||
937 |
where {= title} |
|
938 | ||
cleanup
|
939 |
またL<DBIx::Custom>は文字列の評価をオーバーロードして、C<to_string()> |
added experimental not_exist...
|
940 |
を呼び出すようにしていますので、次のようにしてwhere句を生成することも |
941 |
できます。 |
|
942 | ||
943 |
my $where_clause = "$where"; |
|
944 | ||
945 |
これはSQLの中にwhere句を埋め込むときにとても役立つ機能です。 |
|
946 | ||
make delete() using where ob...
|
947 |
=head3 同一の列名を含む場合 |
added experimental not_exist...
|
948 | |
make delete() using where ob...
|
949 |
タグの中に同一の名前を持つものが存在した場合でも動的に |
950 |
where句を作成することができます。 |
|
added experimental not_exist...
|
951 | |
make delete() using where ob...
|
952 |
たとえば、パラメータとして開始日付と終了日付を受け取ったことを |
953 |
考えてみてください。 |
|
added experimental not_exist...
|
954 | |
make delete() using where ob...
|
955 |
my $param = {start_date => '2010-11-15', end_date => '2011-11-21'}; |
added experimental not_exist...
|
956 | |
make delete() using where ob...
|
957 |
また開始日付と終了日付の片方だけや、どちらも受け取らない場合もあるかもしれません。 |
added experimental not_exist...
|
958 | |
make delete() using where ob...
|
959 |
この場合は次のようなパラメータに変換することで対応することができます。 |
added experimental not_exist...
|
960 | |
make delete() using where ob...
|
961 |
my $p = {date => ['2010-11-15', '2011-11-21']}; |
added experimental not_exist...
|
962 | |
make delete() using where ob...
|
963 |
値が配列のリファレンスになっていることに注目してください。このようにすれば |
964 |
同名の列を含むタグに順番に埋め込むことができます。 |
|
added experimental not_exist...
|
965 | |
make delete() using where ob...
|
966 |
$where->clause( |
967 |
['and', '{> date}', '{< date}'] |
|
968 |
); |
|
969 |
$where->param($p); |
|
970 | ||
971 |
また開始日付が存在しない場合は次のようなデータを作成します。 |
|
972 | ||
973 |
my $p = {date => [$dbi->not_exists, '2011-11-21']}; |
|
974 | ||
975 |
L<DBIx::Custom>のC<not_exists>でDBIx::Custom::NotExistsオブジェクトを |
|
976 |
取得できます。これは対応する値が存在しないことを示すためのものです。 |
|
977 | ||
978 |
また終了日付が存在しない場合は次のようなデータを作成します。 |
|
979 | ||
980 |
my $p = {date => ['2010-11-15']}; |
|
added experimental not_exist...
|
981 | |
make delete() using where ob...
|
982 |
どちらも存在しない場合は次のようなデータを作成します。 |
983 | ||
984 |
my $p = {date => []}; |
|
985 | ||
986 |
少し難しいので一番簡単に作成できるロジックを示しておきます。 |
|
987 | ||
988 |
my @date; |
|
989 |
push @date, exists $param->{start_date} ? $param->{start_date} |
|
990 |
: $dbi->not_exists; |
|
991 |
push @date, $param->{end_date} if exists $param->{end_date}; |
|
992 |
my $p = {date => \@date}; |
|
993 | ||
994 |
=head3 C<select()>との連携 |
|
995 | ||
update pod
|
996 |
L<DBIx::Custom::Where>オブジェクトは |
997 |
C<select()>のC<where>に直接渡すことが |
|
make delete() using where ob...
|
998 |
できます。 |
999 |
|
|
1000 |
my $where = $dbi->where; |
|
1001 |
$where->clause(...); |
|
1002 |
$where->param($param); |
|
1003 |
my $result = $dbi->select(table => 'book', where => $where); |
|
1004 | ||
update pod
|
1005 |
あるいはC<update()>、C<delete()>のwhereに指定することも可能です。 |
make delete() using where ob...
|
1006 | |
1007 |
=head3 C<execute()>との連携 |
|
1008 | ||
1009 |
C<execute()>との連携です。SQLを作成するときに埋め込むことができます。 |
|
1010 | ||
1011 | ||
1012 |
my $where = $dbi->where; |
|
1013 |
$where->clause(...); |
|
1014 |
$where->param($param); |
|
1015 | ||
1016 |
my $sql = <<"EOS" |
|
1017 |
select * from book; |
|
1018 |
$where |
|
1019 |
EOS |
|
added experimental not_exist...
|
1020 | |
make delete() using where ob...
|
1021 |
$dbi->execute($sql, param => $param); |
added experimental not_exist...
|
1022 | |
update pod
|
1023 |
=head2 7. テーブルモデル |
1024 | ||
1025 |
=head3 テーブルオブジェクトの作成 C<table()> |
|
1026 | ||
1027 |
L<DBIx::Custom>はロジックとしてテーブルを中心にすえた |
|
1028 |
モデルの作成を支援します。 |
|
1029 | ||
1030 |
アプリケーションのロジックを記述するときに、そのロジックを |
|
1031 |
データベースのテーブルにすえることは、RDBMSを利用する |
|
1032 |
モデルであれば、コードの重複も少なく |
|
1033 |
わかりやすいものになります。 |
|
1034 | ||
1035 |
テーブルオブジェクトを生成するにはC<table()>を使用します。 |
|
1036 | ||
1037 |
my $table = $dbi->table('book'); |
|
1038 | ||
1039 |
実際にデータベースにテーブルは存在している必要はありません。 |
|
1040 |
これは仮想的なテーブルオブジェクトです。これは |
|
1041 |
L<DBIx::Customm::Table>オブジェクトになります。 |
|
1042 | ||
1043 |
テーブルオブジェクトからはC<insert()>、C<update()>、C<update_all()>、 |
|
1044 |
C<delete()>、C<delete_all()>、C<select()>などのメソッド呼び出すことができます。 |
|
1045 |
L<DBIx::Custom>と異なるところは、C<table>を必ずしも指定する必要が |
|
1046 |
ないということです。 |
|
1047 | ||
1048 |
$table->insert(param => $param); |
|
1049 | ||
1050 |
C<table>の値は自動的にbookに設定されます。 |
|
1051 | ||
1052 |
またテーブルオブジェクトには独自のメソッドを追加することができます。 |
|
1053 | ||
1054 |
$table->method( |
|
1055 |
register => sub { |
|
1056 |
my $self = shift; |
|
1057 |
my $table_name = $self->name; |
|
1058 |
# something |
|
1059 |
}, |
|
1060 |
list => sub { |
|
1061 |
my $self = shift; |
|
1062 |
my $table_name = $self->name; |
|
1063 |
# something |
|
1064 |
} |
|
1065 |
); |
|
1066 | ||
1067 |
メソッドに渡される第一引数はL<DBIx::Custom::Table>オブジェクトです。 |
|
1068 |
C<name()>を使用して、テーブル名を取得することができます。 |
|
1069 | ||
1070 |
このようにして登録したメソッドは直接呼び出すことができます。 |
|
1071 | ||
1072 |
$table->register(...); |
|
1073 |
$table->list(...); |
|
1074 | ||
1075 |
またテーブル専用のメソッドをオーバーライドして作成することもできます。 |
|
1076 | ||
1077 |
$table->method( |
|
1078 |
insert => sub { |
|
1079 |
my $self = shift; |
|
1080 |
|
|
1081 |
$self->base_insert(...); |
|
1082 |
|
|
1083 |
# something |
|
1084 |
} |
|
1085 |
); |
|
1086 | ||
1087 |
もともと存在していたC<insert()>を呼ぶにはC<base_$method>とします。L<DBIx::Custom::Table> |
|
1088 |
のオーバーライドの機能は簡易的なものですが、とても便利です。 |
|
1089 | ||
1090 |
=head2 テーブルで共有のメソッドの登録 |
|
1091 | ||
1092 |
すべてのテーブルでメソッドを共有するにはC<table>メソッドでテーブルを作成する前に、 |
|
1093 |
C<base_table>にメソッドを登録しておきます。 |
|
1094 | ||
1095 |
$dbi->base_table->method( |
|
1096 |
count => sub { |
|
1097 |
my $self = shift; |
|
1098 |
return $self->select(column => ['count(*)']); |
|
1099 |
} |
|
1100 |
); |
|
1101 | ||
1102 |
またテーブルからはL<DBIx::Custom>とL<DBI>のすべてのメソッドを呼び出すことができます。 |
|
1103 | ||
1104 |
# DBIx::Custom method |
|
1105 |
$table->execute($sql); |
|
1106 |
|
|
1107 |
# DBI method |
|
1108 |
$table->begin_work; |
|
1109 |
$table->commit; |
|
1110 | ||
1111 |
=head2 一般的なモデルの構成 |
|
1112 | ||
1113 |
一般的には、L<DBIx::Custom>を継承してコンストラクタの中に、モデルを作成 |
|
1114 |
するのがよいでしょう。 |
|
1115 | ||
1116 |
package MyDBI; |
|
1117 |
|
|
1118 |
use base 'DBIx::Custom'; |
|
1119 |
|
|
1120 |
sub connect { |
|
1121 |
my $self = shift->SUPER::connect(@_); |
|
1122 |
|
|
1123 |
$self->base_table->method( |
|
1124 |
... => sub { ... } |
|
1125 |
); |
|
1126 |
|
|
1127 |
$self->table('book')->method( |
|
1128 |
insert_multi => sub { ... }, |
|
1129 |
... => sub { ... } |
|
1130 |
); |
|
1131 |
|
|
1132 |
$self->table('company')->method( |
|
1133 |
... => sub { ... }, |
|
1134 |
); |
|
1135 |
} |
|
1136 | ||
1137 |
このようにして定義しておけば、次のように利用することができます。 |
|
1138 | ||
1139 |
my $dbi = MyDBI->connect(...); |
|
1140 |
$dbi->table('book')->insert_multi(...); |
|
1141 | ||
1142 |
=head2 8. パフォーマンスの改善 |
|
updated document
|
1143 | |
update pod
|
1144 |
=head3 クエリの作成 |
updated document
|
1145 | |
1146 |
もしC<insert()>メソッドを使用してインサートを実行した場合、 |
|
1147 |
必要なパフォーマンスを得られない場合があるかもしれません。 |
|
1148 |
C<insert()>メソッドは、SQL文とステートメントハンドルを |
|
update pod
|
1149 |
毎回作成するためです。 |
1150 | ||
1151 |
そのような場合は、C<query>オプションを指定することで、 |
|
1152 |
クエリを取得することができます。 |
|
1153 | ||
1154 |
my $query = $dbi->insert(table => 'book', param => $param, query => 1); |
|
1155 | ||
1156 |
またC<create_query()>メソッドを使って任意のSQLのクエリを作成 |
|
1157 |
することもできます。 |
|
updated document
|
1158 | |
1159 |
my $query = $dbi->create_query( |
|
update pod
|
1160 |
"insert into book {insert_param title author};"; |
updated document
|
1161 |
); |
1162 | ||
1163 |
戻り値はL<DBIx::Custom::Query>オブジェクトです。 |
|
1164 |
このオブジェクトはSQL文とパラメータバインド時の列名を |
|
1165 |
保持しています。またステートメントハンドルも保持しています。 |
|
1166 | ||
1167 |
{ |
|
1168 |
sql => 'insert into book (title, author) values (?, ?);', |
|
1169 |
columns => ['title', 'author'], |
|
1170 |
sth => $sth |
|
1171 |
} |
|
1172 | ||
update pod
|
1173 |
クエリオブジェクトを使って繰り返し実行するにはC<execute()>を使用します。 |
updated document
|
1174 |
|
update pod
|
1175 |
my $params = [ |
updated document
|
1176 |
{title => 'Perl', author => 'Ken'}, |
1177 |
{title => 'Good days', author => 'Mike'} |
|
1178 |
]; |
|
1179 |
|
|
update pod
|
1180 |
foreach my $param (@$paramss) { |
1181 |
$dbi->execute($query, table => 'book', param => $input); |
|
updated document
|
1182 |
} |
1183 | ||
1184 |
C<execute>メソッドの第一引数にクエリオブジェトを渡すことができます。 |
|
update pod
|
1185 |
C<insert()>メソッドよりも高速です。 |
updated document
|
1186 | |
update pod
|
1187 |
注意点がいくつかあります。それはパラメータの数は必ず同じでなくてはならない |
1188 |
ということです。最初に3つのパラメータだけを渡したのに、次の実行では |
|
1189 |
二つのパラメータを渡すと予期しない結果になります。それは |
|
1190 |
動的に生成されたSQLに含まれるプレースホルダの数が異なるからです。 |
|
1191 |
またC<execute()>によっては自動的にはフィルタが有効にならないので、 |
|
1192 |
C<table>を指定する必要のあることに注意してください。 |
|
1193 |
本当に必要な場合だけ利用してください。 |
|
updated document
|
1194 | |
update pod
|
1195 |
=head2 9. その他の機能 |
updated document
|
1196 | |
update pod
|
1197 |
=head3 メソッドの登録 |
updated document
|
1198 | |
update pod
|
1199 |
メソッドを登録するにはC<method()>を使用します。 |
updated document
|
1200 | |
update pod
|
1201 |
$dbi->method( |
updated document
|
1202 |
update_or_insert => sub { |
1203 |
my $self = shift; |
|
update pod
|
1204 |
# something |
updated document
|
1205 |
}, |
1206 |
find_or_create => sub { |
|
1207 |
my $self = shift; |
|
update pod
|
1208 |
# something |
updated document
|
1209 |
} |
1210 |
); |
|
1211 | ||
update pod
|
1212 |
<method()>で登録したメソッドは |
updated document
|
1213 |
L<DBIx::Custom>オブジェクトから直接呼び出すことができます。 |
1214 | ||
1215 |
$dbi->update_or_insert; |
|
1216 |
$dbi->find_or_create; |
|
1217 | ||
removed experimental expand
|
1218 |
=head3 結果クラスの変更 |
1219 | ||
1220 |
必要ならば結果クラスを変更することができます。 |
|
1221 | ||
1222 |
package MyResult; |
|
1223 |
use base 'DBIx::Custom::Result'; |
|
1224 |
|
|
1225 |
sub some_method { ... } |
|
updated document
|
1226 | |
removed experimental expand
|
1227 |
1; |
1228 |
|
|
1229 |
package main; |
|
1230 |
|
|
1231 |
use MyResult; |
|
1232 |
|
|
1233 |
my $dbi = DBIx::Custom->connect(...); |
|
1234 |
$dbi->result_class('MyResult'); |
|
updated document
|
1235 | |
removed experimental expand
|
1236 |
=head3 キャッシング |
updated document
|
1237 | |
removed experimental expand
|
1238 |
タグの展開後のSQLはパフォーマンスの理由のためにキャッシュされます。 |
1239 |
これはC<chace>で設定でき、デフォルトではキャッシュを行う設定です。 |
|
updated document
|
1240 | |
removed experimental expand
|
1241 |
$dbi->cache(1); |
updated document
|
1242 | |
removed experimental expand
|
1243 |
キャッシュ方法はC<cache_method>にメソッドを指定することで |
1244 |
変更することができます。 |
|
1245 |
データの保存と取得のためのメソッドを定義します。 |
|
updated document
|
1246 | |
removed experimental expand
|
1247 |
デフォルトでは次のようにメモリ上にキャッシュを行うものになっています。 |
updated document
|
1248 | |
removed experimental expand
|
1249 |
$dbi->cache_method(sub { |
1250 |
sub { |
|
1251 |
my $self = shift; |
|
1252 |
|
|
1253 |
$self->{_cached} ||= {}; |
|
1254 |
|
|
1255 |
if (@_ > 1) { |
|
1256 |
# Set |
|
1257 |
$self->{_cached}{$_[0]} = $_[1] |
|
1258 |
} |
|
1259 |
else { |
|
1260 |
# Get |
|
1261 |
return $self->{_cached}{$_[0]} |
|
1262 |
} |
|
1263 |
} |
|
1264 |
}); |
|
1265 |
|
|
1266 |
第一はL<DBIx::Custom>オブジェクトです。 |
|
1267 |
第二引数はタグの展開される前のSQLです。 |
|
1268 |
第三引数はタグの展開後のSQLです。 |
|
updated document
|
1269 | |
removed experimental expand
|
1270 |
自分で作成する場合は第三引数が存在した場合はキャッシュを設定し、 |
1271 |
存在しなかった場合はキャッシュを取得する実装に |
|
1272 |
してください。 |
|
updated document
|
1273 | |
1274 |
=cut |