詳解Laravel如何透過預先載入優化Model查詢

*文
發布: 2023-03-19 08:16:01
原創
1848 人瀏覽過

本文主要為大家介紹了關於如何利用預加載優化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使用支付寶支付的實例分享

詳解Laravel實作supervisor執行非同步進程的方法

詳解Laravel的任務調度console

##########################

以上是詳解Laravel如何透過預先載入優化Model查詢的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
作者最新文章
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板
關於我們 免責聲明 Sitemap
PHP中文網:公益線上PHP培訓,幫助PHP學習者快速成長!