如何利用预加载优化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 相关文章推荐
数字转英文
Dec 06 PHP
php用户注册页面利用js进行表单验证具体实例
Oct 17 PHP
php使用正则表达式提取字符串中尖括号、小括号、中括号、大括号中的字符串
Apr 05 PHP
destoon实现底部添加你是第几位访问者的方法
Jul 15 PHP
十幅图告诉你什么是PHP引用
Feb 22 PHP
10款实用的PHP开源工具
Oct 23 PHP
PHP利用APC模块实现大文件上传进度条的方法
Oct 29 PHP
PHP版本升级到7.x后wordpress的一些修改及wordpress技巧
Dec 25 PHP
Python中使用django form表单验证的方法
Jan 16 PHP
详解Yii2.0使用AR联表查询实例
Jun 16 PHP
php+redis实现消息队列功能示例
Sep 19 PHP
laravel框架实现为 Blade 模板引擎添加新文件扩展名操作示例
Jan 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
中国收音机工业发展史
2021/03/02 无线电
smarty简单入门实例
2014/11/28 PHP
PHP获取文件夹大小函数用法实例
2015/07/01 PHP
JavaScript的面向对象方法以及差别
2008/03/31 Javascript
javaScript 关闭浏览器 (不弹出提示框)
2010/01/31 Javascript
javascript 基础篇3 类,回调函数,内置对象,事件处理
2012/03/14 Javascript
javascript实现跳转菜单的具体方法
2013/07/05 Javascript
js设置文本框中焦点位置在最后的示例代码(简单实用)
2014/03/04 Javascript
jQuery插件Zclip实现完美兼容个浏览器点击复制内容到剪贴板
2015/04/30 Javascript
Bootstrap实现下拉菜单效果
2016/04/29 Javascript
NodeJs——入门必看攻略
2016/06/27 NodeJs
纯js实现html转pdf的简单实例(推荐)
2017/02/16 Javascript
socket.io与pm2(cluster)集群搭配的解决方案
2017/06/02 Javascript
JavaScript中一些特殊的字符运算
2017/08/17 Javascript
Vue DevTools调试工具的使用
2017/12/05 Javascript
Angular5中提取公共组件之radio list的实例代码
2018/07/10 Javascript
详解vue.js下引入百度地图jsApi的两种方法
2018/07/27 Javascript
用node.js写一个jenkins发版脚本
2019/05/21 Javascript
[02:51]2018年度DOTA2最佳中单位选手-完美盛典
2018/12/17 DOTA
详解Python的Django框架中的templates设置
2015/05/11 Python
浅谈Python的Django框架中的缓存控制
2015/07/24 Python
Python中map,reduce,filter和sorted函数的使用方法
2015/08/17 Python
python字典多键值及重复键值的使用方法(详解)
2016/10/31 Python
Python使用matplotlib实现绘制自定义图形功能示例
2018/01/18 Python
Python读取txt内容写入xls格式excel中的方法
2018/10/11 Python
使用 Python 玩转 GitHub 的贡献板(推荐)
2019/04/04 Python
python 指定源路径来解决import问题的操作
2021/03/04 Python
canvas学习笔记之2d画布基础的实现
2019/02/21 HTML / CSS
canvas 橡皮筋式线条绘图应用方法
2019/02/13 HTML / CSS
德国高尔夫商店:Golfshop.de
2019/06/22 全球购物
高中学生干部学习的自我评价
2014/02/21 职场文书
消防安全承诺书
2014/05/22 职场文书
2015年控辍保学工作总结
2015/05/18 职场文书
讲座开场白台词和结束语
2015/05/29 职场文书
祝寿主持词
2015/07/02 职场文书
《鸡兔同笼》教学反思
2016/02/19 职场文书