首頁 >後端開發 >php教程 >詳解預載優化Laravel Model查詢相關問題

詳解預載優化Laravel Model查詢相關問題

巴扎黑
巴扎黑原創
2017-08-12 11:44:121286瀏覽

這篇文章主要為大家介紹了關於如何利用預加載優化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中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn