使用预加载优化 Laravel Model 查询

file

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

介绍

对象关系映射(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条记录,将会产生多少条查询呢?

通过优化预加载,我们可以避免嵌套关系中的额外查询。

//获取所有文章 - 并预加载文章对应作者及每个作者对应的 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相关的文档已经很全面了,希望额外的实践练习可以帮助您更有信心优化关系查询。


本作品采用《CC 协议》,转载必须注明作者和本文链接
二愣的闲谈杂鱼
本帖由 Summer 于 6年前 加精
《L04 微信小程序从零到发布》
从小程序个人账户申请开始,带你一步步进行开发一个微信小程序,直到提交微信控制台上线发布。
《L02 从零构建论坛系统》
以构建论坛项目 LaraBBS 为线索,展开对 Laravel 框架的全面学习。应用程序架构思路贴近 Laravel 框架的设计哲学。
讨论数量: 19
风吹枫落

我在 blade 循环里面 $item->author->profile; 这个时候产生的查询是怎么样的呢?

6年前 评论
godruoyi

@风吹枫落 如果你在控制器中使用with预加载了你需要的模型,那你在blade中循环$item->author->profile也不会产生查询的。:laughing: :laughing: :laughing: :laughing:

6年前 评论
//获取所有文章 - 并预加载文章对应作者
$posts = App\Post::with('author')->get();//两次查询

//根据每个 `作者` 获取其简介
$posts->map(function ($post) {
    //虽然我们直接通过$author = $post->author不会产生查询,
    //但当调用$author->profile时,每次都会产生一个新查询
    return $post->author->profile;  // 如果 author 正好有个 字段 profile 这里还会产生查询吗?
});
6年前 评论
幽弥狂

得仔细看看 有点看不太懂。。。学习了

6年前 评论
godruoyi

@hookover 直接这样就可以啦!!

User::with('group.attributes');  

//$user->group
//$user->group->attributes
5年前 评论

@godruoyi 是的,有的时候也是忍无可忍用DB写whereIn了.哈哈

4年前 评论
godruoyi

@zwfengwu 没有区别的,with 底层也是调用 wherein(hasMany 关系时) 来实现查询的,只是框架给我们封装了优雅的返回结构,可以让我们直接使用

4年前 评论

其实感觉这种orm的写法跟db的whereIn有什么本质区别吗

4年前 评论
颠倒的玉石

看着觉得懒惰加载和预加载并没有太大的区别呀

4年前 评论
hookover

@godruoyi 谢谢啦,get到了

5年前 评论
hookover

多层预加载怎么做?

User {
      public function group(){
              return $this->beLongTo(Group::class,'gid','gid')
      }
}

Group{
      public function attributes(){
            return $this->hasMany(Attributes::class,'aid','aid');
      }
}

$a = $user->group->attributes(); //某个用户所属分组的所以属性

//如果我在查User表的时候,希望同时预加载Group和Group对应的attributes应该怎么做?

User::with('group')->with('attributes', function($query){
          return $query->with('attributes')
});  
5年前 评论
godruoyi

@Complicated 你说的是哪个骚操作呀

5年前 评论
Complicated

@godruoyi 确定这样不会产生查询??

5年前 评论
godruoyi

@wlight 可以啊

$posts = App\User::with(['posts' => function($query){
    $query->limit(2);
}])->get();

相应的 SQL

select * from `users`  
select * from `posts` where `posts`.`user_id` in ('1', '2', '3', '4', '5', '6', '7', '8', '9', '10') limit 2  
5年前 评论

可以在闭包中限制数目吗?

5年前 评论
wenber

@lybc 在定义关联模式的时候可以设定需要的字段

6年前 评论
godruoyi

@mingyun 我测了一下,不会产生查询

//当author正好有个属性名叫profile
//并且在Author模型定义了profile关联时
public function profile()
{
    return $this->hasOne('App\Models\UserProfile');
}

//这样讲直接返回author属性profile的值,不会产生关联查询
$post->author->profile
6年前 评论
godruoyi

@lybc 你可以用闭包试试

$posts = App\Post::with(['author' => function($query){
    $query->select('id,name');
}])->get();//两次查询
6年前 评论

定义了模型关联时如何只查询其中的某几个字段呢?

6年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!