使用预加载优化Laravel Model查询

godruoyi

原文译自eloquent-eager-loading,简化其前面构造数据部分。php

介绍

对象关系映射(ORM)使数据库的工做变得很是简单。 在以面向对象的方式定义数据库关系时,能够轻松查询相关的模型数据,开发人员可能不会注意底层数据库调用。laravel

下面将经过一些例子,进一步帮助您了解如何优化查询。sql

假设您从数据库收到了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次查询。post

预加载

接下来,若是咱们打算使用关联的模型数据,咱们可使用预加载将该101个查询总数减小到2个查询。 只须要告诉模型你须要什么来加载。以下:优化

//获取已发布的100条文章  - 并预加载文章对应做者
$posts = Post::with('author')->limit(100)->get();//2次查询

$authors = array_map(function($post) {
    // 对做者模型生成查询
    return $post->author->name;//这里讲不在产生查询
}, $posts);

若是你开启了sql日志,你将看到上述预加载将只会产生两条查询:spa

select * from `posts`
select * from `authors` where `authors`.`id` in (?, ?, ?, ?, ?) [1,2,3,4,5]

若是您有多个关联模型,则可使用数组加载它们:日志

$posts = App\Post::with(['author', 'comments'])->get();

接下来咱们从新定义以下关系code

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相关的文档已经很全面了,但愿额外的实践练习能够帮助您更有信心优化关系查询。


图片描述

相关文章
相关标签/搜索