如何利用预加载优化Laravel Model查询详解


Posted in PHP onAugust 11, 2017

前言

本文主要给大家介绍了关于利用预加载优化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,简化其前面构造数据部分。

PHP 相关文章推荐
PHP去除数组中重复的元素并按键名排序函数
Aug 18 PHP
php仿discuz分页效果代码
Oct 02 PHP
用sql命令修改数据表中的一个字段为非空(not null)的语句
Jun 04 PHP
TMDPHP 模板引擎使用教程
Mar 13 PHP
PHP5函数小全(分享)
Jun 06 PHP
PHP mkdir()无写权限的问题解决方法
Jun 19 PHP
php微信公众开发之获取周边酒店信息的方法
Dec 22 PHP
PHP使用curl模拟post上传及接收文件的方法
Mar 04 PHP
php生成酷炫的四个字符验证码
Apr 22 PHP
php投票系统之增加与删除投票(管理员篇)
Jul 01 PHP
PHP中常见的密码处理方式和建议总结
Oct 14 PHP
用Laravel轻松处理千万级数据的方法实现
Dec 25 PHP
PHP实现的自定义图像居中裁剪函数示例【测试可用】
Aug 11 #PHP
Redis在Laravel项目中的应用实例详解
Aug 11 #PHP
PHP验证码无法显示的原因及解决办法
Aug 11 #PHP
php readfile()修改文件上传大小设置
Aug 11 #PHP
浅谈Laravel中的一个后期静态绑定
Aug 11 #PHP
浅谈PHP中new self()和new static()的区别
Aug 11 #PHP
php使用 readfile() 函数设置文件大小大小的方法
Aug 11 #PHP
You might like
php json_encode奇怪问题说明
2011/09/27 PHP
PHP会话控制:Session与Cookie详解
2014/09/27 PHP
PHP将身份证正反面两张照片合成一张图片的代码
2017/04/08 PHP
php无限级评论嵌套实现代码
2018/04/18 PHP
js 获取中文拼音,Select自动匹配字母获取值的代码
2009/09/23 Javascript
按给定几率进行随机抽取的js代码
2010/12/28 Javascript
Javascript合并表格中具有相同内容单元格示例
2013/08/11 Javascript
js 动态加载事件的几种方法总结
2013/12/25 Javascript
jquery事件重复绑定的快速解决方法
2014/01/03 Javascript
DOM基础教程之使用DOM控制表单
2015/01/20 Javascript
通过js获取上传的图片信息(临时保存路径,名称,大小)然后通过ajax传递给后端的方法
2015/10/01 Javascript
Google 地图类型详解及示例代码
2016/08/06 Javascript
用file标签实现多图文件上传预览
2017/02/14 Javascript
详解vue-router 2.0 常用基础知识点之router-link
2017/05/10 Javascript
jQuery Validate 校验多个相同name的方法
2017/05/18 jQuery
利用VS Code开发你的第一个AngularJS 2应用程序
2017/12/15 Javascript
JavaScript实现拖拽功能
2020/02/11 Javascript
Python3.x中自定义比较函数
2015/04/24 Python
Python利用ansible分发处理任务
2015/08/04 Python
Python3.7 dataclass使用指南小结
2019/02/22 Python
python全栈知识点总结
2019/07/01 Python
浅谈Python的方法解析顺序(MRO)
2020/03/05 Python
最新Python idle下载、安装与使用教程图文详解
2020/11/28 Python
大学生自荐信
2013/12/11 职场文书
小学运动会广播稿200字(十二篇)
2014/01/14 职场文书
八一演出活动方案
2014/02/03 职场文书
通知的格式范文
2015/04/27 职场文书
幼儿园秋季开学通知
2015/07/16 职场文书
公司食堂管理制度
2015/08/05 职场文书
反四风问题学习心得体会
2016/01/22 职场文书
《黄山奇石》教学反思
2016/02/18 职场文书
MySQL update set 和 and的区别
2021/05/08 MySQL
Django使用redis配置缓存的方法
2021/06/01 Redis
能让Python提速超40倍的神器Cython详解
2021/06/24 Python
Windows环境下实现批量执行Sql文件
2021/10/05 SQL Server
基于MySql验证的vsftpd虚拟用户
2021/11/07 MySQL