DEVELOPERS BLOGデベロッパーズブログ
- HOME >
- 加藤 正人のデベロッパーズブログ >
- CakePHP の Paginator の sort() で昇順ソート時に NULL 値を最後にしたい
加藤 正人のデベロッパーズブログ
- 氏名
- 加藤 正人
- 役職
- 多分SE
- 血液型
- 秘密
- 出没
- 美味しいもののあるところ
- 特色
- タヒチ大好き。ちょいメタボ。
- 2020/12/15
- CakePHP3 のリダイレクトがうまく行かない例
- 2020/11/12
- Apache Bench
- 2020/10/05
- PhpSpreadsheet でExcel ファイルを読み込む (CakePHP3)
MySQL では NULL 値は昇順ソート時には最先頭、降順ソート時には最後尾になる仕様となっている。(これはデータベースエンジンごとに異なるらしい。)
しかし、この順序を変更したいこともままある。
そのような場合は ORDER 句の先頭に、例えば ISNULL({フィールド名}) ASC を追加することで対応できる。
例: 価格 price フィールドを昇順でソートしたいが、price フィールドが NULL のものは末尾に集めたい場合は次のようにする。
... ORDER BY ISNULL(price) ASC, price ASC ...
さて、ここで問題が1つ。CakePHP の PaginatorHelper::sort() を使用してソート順を指定する場合、sort() メソッドにはソートフィールドを1つしか宣言できない。
いろいろ調べてみたところ、「5丁目通信(仮称) CakePHP:paginateのsortに複数のソート項目を追加する。」という記事が非常に参考になった。
当該モデルの beforeFind() コールバックメソッドで特定の条件の時にのみ、クエリー配列の order 部に array('ISNULL(...)' => 'ASC') を追加する方法だ。これを利用して、
public function beforeFind($query) { if ( @$query['order'] ) { // クエリーにオーダー指定あり foreach ( $query['order'] as $order ) { if ( strtoupper(@$order["{$this->name}.price"]) == 'ASC' ) { // オーダー指定に price フィールドが ASC で指定されている場合は、オーダー配列の先頭に "ISNULL({モデル名}.price) ASC" を追加する array_unshift($query['order'], array("ISNULL({$this->name}.price)" => 'asc')); break; } } } return $query; }
のようにすることで所期の目的を達成することができる。
関連タグ:
CakePHP2
PaginatorHelper
関連エントリー
- 2013/11/09
- CakePHP 2.x の PaginatorHelper::sort の不思議
2020/12/15
2020/11/12
2020/10/05
2020/09/11
2020/08/03