如何利用预加载优化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文件时顶部出现空白影响布局分析原因及解决办法
Jan 11 PHP
比较简单的百度网盘文件直链PHP代码
Mar 24 PHP
php多文件上传功能实现原理及代码
Apr 18 PHP
php 批量生成html,txt文件的实现代码
Jun 26 PHP
destoon数据库表说明汇总
Jul 15 PHP
set_exception_handler函数在ThinkPHP中的用法
Oct 31 PHP
PHP PDOStatement对象bindpram()、bindvalue()和bindcolumn之间的区别
Nov 20 PHP
php强制更新图片缓存的方法
Feb 11 PHP
Thinkphp关闭缓存的方法
Jun 26 PHP
PHP用户管理中常用接口调用实例及解析(含源码)
Mar 09 PHP
php微信公众号开发之校园图书馆
Oct 20 PHP
PHP的Trait机制原理与用法分析
Oct 18 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
Syphon 虹吸式咖啡壶冲煮–拨动法
2021/03/03 冲泡冲煮
php设计模式之适配器模式实例分析【星际争霸游戏案例】
2020/04/07 PHP
JCalendar 日历控件 v1.0 beta[兼容IE&Firefox] 有文档和例子
2007/05/30 Javascript
javascript 另一种图片滚动切换效果思路
2012/04/20 Javascript
IE下window.onresize 多次调用与死循环bug处理方法介绍
2013/11/12 Javascript
jQuery实现的导航条切换可显示隐藏
2014/10/22 Javascript
js前端日历控件(悬浮、拖拽、自由变形)
2017/03/02 Javascript
JS异步文件上传(兼容IE8+)
2017/04/02 Javascript
使用Node.js搭建静态资源服务详细教程
2017/08/02 Javascript
JS实现随机生成10个手机号的方法示例
2018/12/07 Javascript
说说Vuex的getters属性的具体用法
2019/04/15 Javascript
laypage.js分页插件使用方法详解
2019/07/27 Javascript
JavaScript Tab菜单实现过程解析
2020/05/13 Javascript
taro 实现购物车逻辑的实例代码
2020/06/05 Javascript
vue 解决uglifyjs-webpack-plugin打包出现报错的问题
2020/08/04 Javascript
Vue向后台传数组数据,springboot接收vue传的数组数据实例
2020/11/12 Javascript
[02:22:36]《加油!DOTA》总决赛
2014/09/19 DOTA
[04:51]TI10典藏宝瓶Ⅱ外观视频展示
2020/08/15 DOTA
Python实现简单网页图片抓取完整代码实例
2017/12/15 Python
利用python修改json文件的value方法
2018/12/31 Python
python按键按住不放持续响应的实例代码
2019/07/17 Python
使用Python实现画一个中国地图
2019/11/23 Python
pyqt5 textEdit、lineEdit操作的示例代码
2020/08/12 Python
Python编写memcached启动脚本代码实例
2020/08/14 Python
Java工程师面试集锦之Spring框架
2013/06/16 面试题
毕业生自我推荐
2013/11/04 职场文书
大学生涯自我鉴定
2014/01/16 职场文书
集体备课反思
2014/02/12 职场文书
小组合作学习反思
2014/02/18 职场文书
优秀党员学习焦裕禄精神思想汇报范文
2014/09/10 职场文书
大学生自荐信怎么写
2015/03/26 职场文书
色戒观后感
2015/06/12 职场文书
大学生暑期实践报告
2015/07/13 职场文书
2019朋友新婚祝福语精选
2019/10/10 职场文书
Python基础之操作MySQL数据库
2021/05/06 Python
探究Mysql模糊查询是否区分大小写
2021/06/11 MySQL