Laravel Query Builder 复杂查询案例:子查询实现分区查询 partition by
案例
案例:[Laravel 在文章列表中附帶上前10條評論?][1],在獲取文章列表時同時把每個文章的前10條評論一同查詢出來。
這是典型分區查詢案例,需要根據 comments 表中的 post_id 字段進行分區,同時根據條件進行排序,把符合條件的前 N 條是數據取出來。
在其他數據庫(Oracle, SQL Server,Vertica) 包含了 row_number partition by 這樣的函數,能夠比較容易的實現。
比如在 SQL Server 中:
SELECT * FROM ( SELECT *, row_number() OVER (partition by post_id ORDER BY created_at desc) rank FROM comments where post_id in (1,2,3,4,5) ) b where rand < 11; 復制代碼在 mysql 中要復雜一些,我們先來看看上面案例中實現需求的幾種解決辦法。
解決辦法
方法1:
在 blade 中要顯示評論數據的地方 post->comments()->limit(10)
問題:如果取了 20 條 Post 數據,就會有 20 條取 comments 的 sql 語句,會造成執行的 sql 語句過多。
不是非常可取,主要問題會造成 SQL 語句過多,對數據庫服務器產生壓力,不過這里可以使用緩存來改進,但是不在本文章討論范圍里。
方法2:
直接通過 with 把 Post 的所有 comments 數據都取出來,在 blade 中 post->comments->take(10)
問題:Laravel 會預先把文章所有的評論數據查詢出來,如果文章的評論數據非常多,可能會造成內存泄漏。
方法3:
$posts = Post::paginate(15);$postIds = $posts->pluck('id')->all();//找出符合條件的 comments ,同時定義 @post, @rank 變量,這里沒有用 all,get 等函數,此時并不會執行 SQL 語句。 $sub = Comment::whereIn('post_id',$postIds)->select(DB::raw('*,@post := NULL ,@rank := 0'))->orderBy('post_id');//把上面構造的 sql 查詢作為子表進行查詢,根據 post_id 進行分區的同時 @rank 變量不斷+1 $sub2 = DB::table( DB::raw("({$sub->toSql()}) as b") )->mergeBindings($sub->getQuery())->select(DB::raw('b.*,IF (@post = b.post_id ,@rank :=@rank + 1 ,@rank := 1) AS rank,@post := b.post_id'));//取出符合條件的前10條comment $commentIds = DB::table( DB::raw("({$sub2->toSql()}) as c") )->mergeBindings($sub2)->where('rank','<',11)->select('c.id')->pluck('id')->toArray();$comments = Comment::whereIn('id',$commentIds)->get();$posts = $posts->each(function ($item, $key) use ($comments) {$item->comments = $comments->where('post_id',$item->id); }); 復制代碼會產生三條sql
select * from `posts` limit 15 offset 0;select `c`.`id` from (select b.*,IF ( @post = b.post_id ,@rank :=@rank + 1 ,@rank := 1 ) AS rank, @post := b.post_id from (select *,@post := NULL ,@rank := 0 from `comments` where `post_id` in ('2', '3', '4', '5', '6', '7', '8', '9', '10', '11', '12', '13', '14', '15', '16') order by `post_id` asc) as b) as c where `rank` < '11';select * from `comments` where `id` in ('180', '589', '590', '3736'); 復制代碼知識點
為什么不直接用原生 SQL 語句來實現?
這里之所以堅持使用 Laravel Query Builder 來實現,可以有效防止 SQL 注入,并且和 ORM 的 Model 對象關聯起來。
如果還有更多類似這種復雜的需求,歡迎聯系我 : )
討論交流
總結
以上是生活随笔為你收集整理的Laravel Query Builder 复杂查询案例:子查询实现分区查询 partition by的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: HP LoadRunner 12.02
- 下一篇: HeartBeat Install