如何利用预加载优化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 相关文章推荐
动态网站web开发 PHP、ASP还是ASP.NET
Oct 09 PHP
一个简易需要注册的留言版程序
Oct 09 PHP
从手册去理解分析PHP session机制
Jul 17 PHP
PHP表单提交表单名称含有点号(.)则会被转化为下划线(_)
Dec 14 PHP
php加密解密函数authcode的用法详细解析
Oct 28 PHP
PHP执行Curl时报错提示CURL ERROR: Recv failure: Connection reset by peer的解决方法
Jun 26 PHP
ThinkPHP权限认证Auth实例详解
Jul 22 PHP
PHP生成和获取XML格式数据的方法
Mar 04 PHP
php技巧小结【推荐】
Jan 19 PHP
PHP curl 或 file_get_contents 获取需要授权页面的方法
May 05 PHP
Laravel框架基于中间件实现禁止未登录用户访问页面功能示例
Jan 17 PHP
使用PHP开发留言板功能
Nov 19 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文件读写操作之文件读取方法详解
2011/01/13 PHP
php+ajax简单实现全选删除的方法
2016/12/06 PHP
Laravel 模型关联基础教程详解
2019/09/17 PHP
Three.js源码阅读笔记(物体是如何组织的)
2012/12/27 Javascript
js网页中的(运行代码)功能实现思路
2013/02/04 Javascript
Jquery easyUI 更新行示例
2014/03/06 Javascript
了不起的node.js读书笔记之node.js中的特性
2014/12/22 Javascript
PHP实现的各种中文编码转换类分享
2015/01/23 Javascript
js实现宇宙星空背景效果的方法
2015/03/03 Javascript
Ajax清除浏览器js、css、图片缓存的方法
2015/08/06 Javascript
jQuery实现带延时功能的水平多级菜单效果【附demo源码下载】
2016/09/21 Javascript
jQuery中$.ajax()方法参数解析
2016/10/22 Javascript
本地Bootstrap文件字体图标引入却无法显示问题的解决方法
2020/04/18 Javascript
node.js入门学习之url模块
2017/02/25 Javascript
鼠标拖动改变DIV等网页元素的大小的实现方法
2017/07/06 Javascript
使用taro开发微信小程序遇到的坑总结
2019/04/08 Javascript
Vue.js中的extend绑定节点并显示的方法
2019/06/20 Javascript
在Angular项目使用socket.io实现通信的方法
2021/01/05 Javascript
[01:11]辉夜杯战队访谈宣传片—CDEC.Y
2015/12/26 DOTA
使用Python脚本将绝对url替换为相对url的教程
2015/04/24 Python
Python cx_freeze打包工具处理问题思路及解决办法
2016/02/13 Python
django认证系统实现自定义权限管理的方法
2018/07/16 Python
深入理解Python中的 __new__ 和 __init__及区别介绍
2018/09/17 Python
Python中__repr__和__str__区别详解
2019/11/07 Python
Python测试Kafka集群(pykafka)实例
2019/12/23 Python
详解纯CSS3制作的20种loading动效
2017/07/05 HTML / CSS
纯css3无js实现的Android Logo(有简单动画)
2013/01/21 HTML / CSS
纯CSS改变webkit内核浏览器的滚动条样式
2014/04/17 HTML / CSS
顺丰快递Java软件工程师面试题
2015/07/31 面试题
产品质量承诺书范文
2014/03/27 职场文书
供应链金融服务方案
2014/05/25 职场文书
运动员口号
2014/06/09 职场文书
爱心捐助活动总结
2015/05/09 职场文书
新娘婚礼答谢词
2015/09/29 职场文书
机关单位2016年创先争优活动总结
2016/04/05 职场文书
MySql数据库 查询时间序列间隔
2022/05/11 MySQL