這篇文章主要為大家介紹了關於如何利用預加載優化Laravel Model查詢的相關資料,文章透過範例程式碼介紹的非常詳細,對大家的學習或工作具有一定的參考學習價值,需要的朋友們下面跟著小編來一起學習學習吧。
前言
本文主要為大家介紹了關於利用預載優化Laravel Model查詢的相關內容,分享出來供大家參考學習,話不多說了,來一起看看詳細的介紹:
#介紹
物件關係映射(ORM)使資料庫的工作變得非常簡單。 在以物件導向的方式定義資料庫關係時,可以輕鬆查詢相關的模型數據,開發人員可能不會注意到底層資料庫呼叫。
以下將透過一些例子,進一步幫助您了解如何最佳化查詢。
假設您從資料庫收到了100個對象,並且每個記錄都有1個關聯模型(即belongsTo)。 預設使用ORM將產生101個查詢; 如下所示:
//获取已发布的100条文章 $posts = Post::limit(100)->get(); //一次查询 $authors = array_map(function($post) { // 对作者模型生成查询 return $post->author->name; }, $posts);
我們在查詢時沒有告訴Post模型,我們還需要所有的作者,所以每次從單個Post模型實例取得作者的名字時,都會發生單獨的查詢。
array_maps時發生100次查詢,加上先前一次查詢,累計產生101次查詢。
預先載入
#接下來,如果我們打算使用關聯的模型數據,我們可以使用預先載入將該101個查詢總數減少到2個查詢。 只需要告訴模型你需要什麼來載入。如下:
//获取已发布的100条文章 - 并预加载文章对应作者 $posts = Post::with('author')->limit(100)->get();//2次查询 $authors = array_map(function($post) { // 对作者模型生成查询 return $post->author->name;//这里讲不在产生查询 }, $posts);
如果你開啟了sql日誌,你會看到上述預先載入只會產生兩個查詢:
select * from `posts` select * from `authors` where `authors`.`id` in (?, ?, ?, ?, ?) [1,2,3,4,5]
如果您有多個關聯模型,則可以使用陣列來載入它們:
$posts = App\Post::with(['author', 'comments'])->get();
接下來我們重新定義如下關係
Post -> belongsTo -> Author //每个文章只属于一个用户 Author -> hasMany -> Post //每个用户拥有多个文章 Author -> hasOne -> Profile //每个用户只有一个简介
考慮下述情況:取得已發布文章所屬作者的個人簡介。
//获取所有文章 - 并预加载文章对应作者 $posts = App\Post::with('author')->get();//两次查询 //根据每个 `作者` 获取其简介 $posts->map(function ($post) { //虽然我们直接通过$author = $post->author不会产生查询, //但当调用$author->profile时,每次都会产生一个新查询 return $post->author->profile; });
假設上述App\Post::with('author')->get()
有100筆記錄,將會產生多少條查詢呢?
透過優化預加載,我們可以避免嵌套關係中的額外查詢。
//获取所有文章 - 并预加载文章对应作者及每个作者对应de profile $posts = App\Post::with('author.profile')->get();//三次查询 $posts->map(function ($post) { //不在产生新查询 return $post->author->profile; });
你可以開啟你的sql日誌看到對應的三條查詢。
select * from `posts` select * from `authors` where `authors`.`id` in (?, ?, ?, ?, ?) [.....] select * from `profiles` where `profiles`.`author_id` in (?, ?, ?, ?, ?) [.....]
懶惰載入
#有時候您可能只需要根據條件收集相關聯的模型。 在這種情況下,您可以懶惰地呼叫相關資料的其他查詢:
$posts = App\Post::all();//一次查询 $posts->load('author.profile');//两次查询 $posts->map(function ($post) { //不在产生新查询 return $post->author->profile; });
查看您的sql日誌,總共看到三個查詢,但只有呼叫$ posts->load()時才會顯示。
結論
希望您更加了解有關加載型號的更多信息,並了解其在更深層次上的工作原理。 Laravel相關的文件已經很全面了,希望額外的實作練習可以幫助您更有信心優化關係查詢。
總結
原文譯自eloquent-eager-loading,簡化其前面建構資料部分。
以上是詳解預載優化Laravel Model查詢相關問題的詳細內容。更多資訊請關注PHP中文網其他相關文章!