如何利用预加载优化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 FOR MYSQL 代码生成助手(根据Mysql里的字段自动生成类文件的)
Jul 23 PHP
支持中文字母数字、自定义字体php验证码代码
Feb 27 PHP
zf框架的校验器使用使用示例(自定义校验器和校验器链)
Mar 13 PHP
YII路径的用法总结
Jul 09 PHP
php使用cookie保存登录用户名的方法
Jan 26 PHP
php基于curl扩展制作跨平台的restfule 接口
May 11 PHP
3种方法轻松处理php开发中emoji表情的问题
Jul 18 PHP
功能强大的php分页函数
Jul 20 PHP
功能强大的php文件上传类
Aug 29 PHP
浅谈PHP表单提交(POST&GET&URL编/解码)
Apr 03 PHP
Laravel 5.5官方推荐的Nginx配置学习教程
Oct 06 PHP
php适配器模式简单应用示例
Oct 23 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使用正则表达式获取图片url的方法
2015/01/16 PHP
PHP 实现页面静态化的几种方法
2017/07/23 PHP
JavaScript中的Location地址对象
2008/01/16 Javascript
javaScript NameSpace 简单说明介绍
2013/07/18 Javascript
调用innerHTML之后onclick失效问题的解决方法
2014/01/28 Javascript
js中匿名函数的创建与调用方法分析
2014/12/19 Javascript
jQuery中:text选择器用法实例
2015/01/03 Javascript
js实现浏览器窗口大小被改变时触发事件的方法
2015/02/02 Javascript
jquery实现在光标位置插入内容的方法
2015/02/05 Javascript
JQuery显示隐藏页面元素的方法总结
2015/04/16 Javascript
微信小程序 时间格式化(util.formatTime(new Date))详解
2016/11/16 Javascript
详解Vue2中组件间通信的解决全方案
2017/07/28 Javascript
基于Vue框架vux组件库实现上拉刷新功能
2017/11/28 Javascript
vue实现搜索过滤效果
2019/05/28 Javascript
vue中echarts的用法及与elementui-select的协同绑定操作
2020/11/17 Vue.js
JavaScript canvas实现跟随鼠标移动小球
2021/02/09 Javascript
Python运行的17个时新手常见错误小结
2012/08/07 Python
使用python Django做网页
2013/11/04 Python
pyenv命令管理多个Python版本
2017/03/26 Python
Python使用Matplotlib实现雨点图动画效果的方法
2017/12/23 Python
opencv python 傅里叶变换的使用
2018/07/21 Python
Python爬虫PyQuery库基本用法入门教程
2018/08/04 Python
Python设计模式之职责链模式原理与用法实例分析
2019/01/11 Python
浅谈python opencv对图像颜色通道进行加减操作溢出
2020/06/03 Python
Python pickle模块常用方法代码实例
2020/10/10 Python
10个很棒的 CSS3 开发工具 推荐
2011/05/16 HTML / CSS
澳大利亚巧克力花束和礼品网站:Tastebuds
2019/03/15 全球购物
Andrew Marc官网:设计师外套的领先制造商
2019/10/30 全球购物
10条PHP编程习惯
2014/05/26 面试题
.net面试题
2016/09/17 面试题
高中考试作弊检讨书
2014/01/14 职场文书
大学专科求职信
2014/07/02 职场文书
公司委托书怎么写
2014/08/02 职场文书
幼儿园2014年度工作总结
2014/11/10 职场文书
人事行政助理岗位职责
2015/04/11 职场文书
使用Python解决图表与画布的间距问题
2022/04/11 Python