Laravel模型关联预加载技巧分享
本文深入探讨了 Laravel 模型预加载的优化技巧,旨在解决传统 `$with` 属性可能导致的性能问题。针对模型关联关系并非普遍存在的情况,文章提出利用 Laravel 模型事件中的 `retrieved` 事件,实现按需的条件预加载。即仅当特定条件满足时,才动态加载相关联的数据,从而避免不必要的数据库查询,提升应用程序的性能和资源利用率。通过移除 `$with` 属性中的默认预加载,并在 `boot` 方法中监听 `retrieved` 事件,可以根据模型属性值灵活地加载关联关系,使 Laravel 应用更加健壮和高效。本文不仅提供了详细的实现步骤,还分析了其优势与注意事项,帮助开发者更好地理解和应用这一技巧。

1. 理解 Laravel 预加载与潜在性能问题
在 Laravel Eloquent 中,预加载(Eager Loading)是解决 N+1 查询问题的关键技术。通过在查询时使用 with() 方法或在模型中定义 $with 属性,可以一次性加载所有相关联的数据,避免在循环中对每个模型实例单独执行查询。
例如,一个 User 模型可能关联 Domain 和 BusinessUnits:
class User extends Authenticatable
{
// ...
protected $with = [
'domain',
'BusinessUnits'
];
public function BusinessUnits()
{
return $this->belongsToMany(BusinessUnit::class, 'users_business_units_pivot');
}
public function Domain()
{
return $this->belongsTo(Domain::class);
}
}这种设置方式的优点是简单直接,无论何时查询 User 模型,domain 和 BusinessUnits 都会被自动预加载。然而,当某些关联关系并非对所有模型实例都存在时,这种无差别预加载会导致性能问题。例如,如果只有特定类型的用户(如“客户”)才拥有 domain_id,而其他用户(如“员工”)的 domain_id 为空,那么对“员工”用户预加载 domain 和 BusinessUnits 就会产生不必要的数据库查询,即使这些查询的结果集为空。
尝试在 $with 属性中直接使用条件逻辑,例如:
protected $with = [
(!$this->domain_id) ? 'domain' : null,
(!$this->domain_id) ? 'BusinessUnits' : null
];这种做法会导致“Constant expression contains invalid operations”错误,因为 $with 属性的定义必须是一个常量表达式,不能包含运行时才能确定的变量或对象属性。
2. 利用模型事件实现条件预加载
为了解决上述问题,我们可以在模型被检索(retrieved)之后,根据模型的特定属性值来判断是否需要加载关联关系。Laravel 提供了丰富的模型事件,其中 retrieved 事件在模型从数据库中取出后触发,是执行此类条件逻辑的理想时机。
实现步骤如下:
步骤一:移除 $with 属性中的默认预加载
首先,从 User 模型中的 $with 属性中移除 domain 和 BusinessUnits,以避免无条件预加载:
class User extends Authenticatable
{
// ...
// protected $with = [
// 'domain',
// 'BusinessUnits'
// ]; // 移除或注释掉这两行
// ...
}步骤二:在 boot 方法中监听 retrieved 事件
在 User 模型的 boot 静态方法中,注册一个 retrieved 事件监听器。boot 方法是 Eloquent 模型初始化时调用的,非常适合注册模型事件。
class User extends Authenticatable implements HasMedia
{
// ... 其他 use 语句和属性
/**
* 模型启动时执行的方法。
*
* @return void
*/
protected static function boot()
{
parent::boot(); // 必须调用父类的 boot 方法
// 监听模型从数据库中取出(retrieved)事件
self::retrieved(function ($model) {
// 检查 domain_id 是否不为空
if ($model->domain_id !== null) {
// 如果 domain_id 不为空,则按需加载 'domain' 和 'BusinessUnits' 关联关系
$model->load('domain', 'BusinessUnits');
}
});
}
// ... 其他方法和关联关系定义
}代码解析:
- parent::boot();: 这是非常重要的一步,确保父类 Authenticatable 的 boot 方法也被执行,否则可能会导致一些内置功能失效。
- self::retrieved(function ($model) { ... });: 这行代码注册了一个匿名函数作为 retrieved 事件的监听器。当 User 模型实例从数据库中被检索出来时,这个匿名函数就会被调用,并将当前模型实例作为参数 $model 传递进来。
- if ($model->domain_id !== null) { ... }: 在监听器内部,我们检查当前 $model 实例的 domain_id 属性。只有当 domain_id 不为 null 时,才执行预加载逻辑。
- $model->load('domain', 'BusinessUnits');: load() 方法用于在模型实例已经被检索出来之后,动态地加载指定的关联关系。它会执行相应的数据库查询并将关联数据填充到模型实例中。
3. 优势与注意事项
优势:
- 性能优化: 显著减少了不必要的数据库查询,特别是对于那些不具备特定关联关系的模型实例。
- 资源利用率提高: 避免了加载和处理冗余数据,降低了内存消耗。
- 代码清晰: 将条件逻辑从模型属性定义中分离出来,使模型定义更加简洁。
- 灵活性: 这种方法可以应用于任何复杂的条件,而不仅仅是基于单个字段的判断。
注意事项:
- 适用场景: 这种方法最适用于模型实例已经被取出,并且你需要在其上进行操作时。如果你需要在查询构建阶段就进行条件预加载(例如,只对特定 scope 的查询进行预加载),那么 with() 方法的闭包形式可能更合适:
User::whereNotNull('domain_id')->with(['domain', 'BusinessUnits'])->get(); // 或者结合作用域 User::client()->with(['domain', 'BusinessUnits'])->get();然而,本文讨论的场景是无论通过何种方式检索模型,只要 domain_id 存在,就自动预加载,这正是 retrieved 事件的用武之地。
- N+1 问题变体: 尽管解决了不必要的预加载,但如果 domain_id 不为空的用户数量很多,load() 方法仍然可能导致 N+1 查询问题,因为它会在每次 retrieved 事件中执行一次查询。对于大量满足条件的用户,Laravel 的 with() 方法在构建查询时会生成单个 JOIN 或 IN 子句来加载所有相关数据,效率更高。 然而,需要明确的是,本教程解决的是“不必要的预加载”问题,而不是“预加载本身的效率”问题。 如果你查询的是单个 User 模型,或者少量 User 模型,load() 方法的开销是可接受的,并且它能确保只有满足条件的用户才触发关联查询。
- 序列化: 使用 load() 方法加载的关联关系会像通过 with() 预加载一样,在模型被转换为数组或 JSON 时自动包含进去。
4. 总结
通过在 Laravel 模型中使用 retrieved 事件,我们可以实现基于条件的按需预加载,有效避免了无差别预加载带来的性能开销。这种方法提供了一种灵活且高效的策略,尤其适用于那些关联关系并非对所有模型实例都普遍存在的场景,从而使我们的 Laravel 应用更加健壮和高效。
今天带大家了解了的相关知识,希望对你有所帮助;关于文章的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~
京东金条怎么开通?快速找到入口方法
- 上一篇
- 京东金条怎么开通?快速找到入口方法
- 下一篇
- PHP实现Ajax数据交互教程
-
- 文章 · php教程 | 31分钟前 |
- PHP路由参数传递与call_user_func_array用法详解
- 115浏览 收藏
-
- 文章 · php教程 | 37分钟前 |
- PHP循环中数据库查询优化方法
- 374浏览 收藏
-
- 文章 · php教程 | 52分钟前 | docker Nginx dockercompose php-fpm PHP多版本
- Docker部署PHP多版本共存教程详解
- 355浏览 收藏
-
- 文章 · php教程 | 58分钟前 |
- PHP数组高效操作与常用函数解析
- 438浏览 收藏
-
- 文章 · php教程 | 1小时前 |
- PHP不区分大小写查找数组值的方法
- 432浏览 收藏
-
- 文章 · php教程 | 1小时前 |
- PDO占位符错误怎么解决?
- 369浏览 收藏
-
- 文章 · php教程 | 1小时前 | session $_COOKIE setcookie() PHPCookie Cookie安全
- PHP设置与获取Cookie的完整教程
- 407浏览 收藏
-
- 文章 · php教程 | 1小时前 |
- PHP正则替换:精准控制重复模式次数
- 275浏览 收藏
-
- 文章 · php教程 | 2小时前 | php GD库 getimagesize() ImageMagick 图片尺寸
- PHP获取图片尺寸的实用方法
- 408浏览 收藏
-
- 文章 · php教程 | 2小时前 |
- Yii2ActiveFormJS验证与按钮联动技巧
- 382浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3203次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3416次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3446次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4554次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3824次使用
-
- PHP技术的高薪回报与发展前景
- 2023-10-08 501浏览
-
- 基于 PHP 的商场优惠券系统开发中的常见问题解决方案
- 2023-10-05 501浏览
-
- 如何使用PHP开发简单的在线支付功能
- 2023-09-27 501浏览
-
- PHP消息队列开发指南:实现分布式缓存刷新器
- 2023-09-30 501浏览
-
- 如何在PHP微服务中实现分布式任务分配和调度
- 2023-10-04 501浏览

