如何利用预加载优化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 变量定义方法
Jun 14 PHP
PHP安全性漫谈
Jun 28 PHP
php二分查找二种实现示例
Mar 12 PHP
windows平台中配置nginx+php环境
Dec 06 PHP
php使用escapeshellarg时中文被过滤的解决方法
Jul 10 PHP
基于jQueryUI和Corethink实现百度的搜索提示功能
Nov 09 PHP
php 基础函数
Feb 10 PHP
微信公众号开发之获取位置信息php代码
Jun 13 PHP
PHP中检查isset()和!empty()函数的必要性
Feb 13 PHP
Thinkphp自定义生成缩略图尺寸的方法
Aug 05 PHP
laravel执行php artisan migrate报错的解决方法
Oct 09 PHP
PHP7创建销毁session的实例方法
Feb 03 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
关于更改Zend Studio/Eclipse代码风格主题的介绍
2013/06/23 PHP
解决laravel5.4下的group by报错的问题
2019/10/16 PHP
两种方法实现文本框输入内容提示消失
2013/03/17 Javascript
当鼠标滑过文本框自动选中输入框内容的JS代码分享
2013/11/26 Javascript
一个简单的全屏图片上下打开显示网页效果示例
2014/07/08 Javascript
node.js中的favicon.ico请求问题处理
2014/12/15 Javascript
jQuery fancybox在ie浏览器下无法显示关闭按钮的解决办法
2016/02/19 Javascript
全面解析JS字符串和正则表达式中的match、replace、exec等函数
2016/07/01 Javascript
详解如何较好的使用js
2016/12/16 Javascript
js实现点击切换checkbox背景图片的简单实例
2017/05/08 Javascript
使用Vue完成一个简单的todolist的方法
2017/12/01 Javascript
form表单数据封装成json格式并提交给服务器的实现方法
2017/12/14 Javascript
详解Vue项目编译后部署在非网站根目录的解决方案
2018/04/26 Javascript
vue表单自定义校验规则介绍
2018/08/28 Javascript
js实现消灭星星(web简易版)
2020/03/24 Javascript
vue+element实现图片上传及裁剪功能
2020/06/29 Javascript
Vue2.x和Vue3.x的双向绑定原理详解
2020/11/05 Javascript
[33:09]完美世界DOTA2联赛循环赛 Forest vs DM BO2第二场 10.29
2020/10/29 DOTA
python脚本实现xls(xlsx)转成csv
2016/04/10 Python
Python字典,函数,全局变量代码解析
2017/12/18 Python
解决csv.writer写入文件有多余的空行问题
2018/07/06 Python
Python OpenCV处理图像之图像直方图和反向投影
2018/07/10 Python
Python脚本利用adb进行手机控制的方法
2019/07/08 Python
Python调用scp向服务器上传文件示例
2019/12/22 Python
Pytorch技巧:DataLoader的collate_fn参数使用详解
2020/01/08 Python
Python图像阈值化处理及算法比对实例解析
2020/06/19 Python
CSS3之背景尺寸Background-size使用介绍
2013/10/14 HTML / CSS
ALLSAINTS英国官网:伦敦新锐潮流品牌
2016/09/19 全球购物
法国珠宝店:CLEOR
2017/01/29 全球购物
英国派对礼服和连衣裙购物网站:TFNC London
2018/07/07 全球购物
完美的中文自荐信
2014/05/24 职场文书
2015教师节通讯稿
2015/07/20 职场文书
2016小学优秀教师先进事迹材料
2016/02/26 职场文书
2016年村党支部公开承诺书
2016/03/24 职场文书
商业计划书之服装
2019/09/09 职场文书
JavaWeb Servlet开发注册页面实例
2022/04/11 Java/Android