0

0

Laravel关系查询性能瓶颈?biiiiiiigmonster/hasin助你告别WHEREEXISTS慢查询!

碧海醫心

碧海醫心

发布时间:2025-11-25 11:08:02

|

146人浏览过

|

来源于php中文网

原创

laravel关系查询性能瓶颈?biiiiiiigmonster/hasin助你告别whereexists慢查询!

可以通过一下地址学习composer学习地址

在日常的 Laravel 开发中,我们经常需要查询那些拥有特定关联模型的记录。例如,找出所有发布过文章的用户,或者查询所有拥有评论的帖子。这时候,Laravel 强大的 Eloquent ORM 提供了 has() 方法,用起来简直不要太方便:

// 找出所有发布过文章的用户
User::has('posts')->get();

初看起来,这简直是完美的解决方案。然而,随着项目数据的不断增长,尤其是当 User 表(外部表)的数据量变得庞大时,你可能会发现应用的某些页面加载速度开始变慢,数据库的 CPU 使用率飙升,用户体验直线下降。这背后的“罪魁祸首”,往往就是 Laravel 默认 has() 方法所采用的底层 SQL 实现——WHERE EXISTS

WHERE EXISTS 的性能陷阱

当我们执行 User::has('posts')->get() 时,Laravel 在底层会生成类似这样的 SQL 语句:

SELECT * FROM `users` WHERE EXISTS (SELECT * FROM `posts` WHERE `users`.`id` = `posts`.`user_id`)

WHERE EXISTS 的工作原理是:它会遍历 users 表中的每一条记录,然后对每一条记录都执行一次子查询(SELECT * FROM posts WHERE users.id = posts.user_id)。如果 users 表有百万级数据,那么这个子查询就会被执行百万次!尽管数据库会尝试优化,但这种“循环查询”的模式,在外部表数据量巨大时,性能瓶颈是显而易见的。

我曾在一个电商项目中遇到过类似的问题。我们需要筛选出所有购买过特定商品的客户,客户表和订单表都非常庞大。使用 has('orders.items') 这样的查询,页面响应时间从毫秒级直接飙升到数秒,甚至导致请求超时。尝试优化索引、调整查询顺序等常规手段后,效果依然不理想,这让我一度陷入困境。难道要手写复杂的 JOIN 语句来替代 Eloquent 的优雅吗?

biiiiiiigmonster/hasin:性能优化的救星

就在我为性能问题焦头烂额之际,我发现了 biiiiiiigmonster/hasin 这个 Composer 包。它提供了一个基于 WHERE IN 语法的关系查询实现,旨在替代 has 在某些业务场景下的 WHERE EXISTS 实现,从而获得更高的性能。

它的核心思想很简单:WHERE EXISTS 替换为 WHERE IN

我们来看看 WHERE IN 的 SQL 结构:

SELECT * FROM `users` WHERE `users`.`id` IN (SELECT `posts`.`user_id` FROM `posts`)

WHERE IN 的工作方式则大不相同:它会首先执行子查询 SELECT posts.user_id FROM posts,得到一个用户 ID 列表。然后,主查询只需要判断 users.id 是否在这个列表中。这种方式通常会先查询内部表,然后将内部表的结果与外部表进行匹配,尤其当外部表(users)数据量很大时,其效率远高于 WHERE EXISTS

安装与使用

biiiiiiigmonster/hasin 集成到你的 Laravel 项目中非常简单,只需通过 Composer 安装即可:

# 根据你的 Laravel 版本选择对应的安装命令
composer require biiiiiigmonster/hasin:^5.0 # For Laravel 12
composer require biiiiiiiigmonster/hasin:^4.0 # For Laravel 11
# ...以此类推,或查看文档选择适合你Laravel版本的命令

安装完成后,你就可以在 Eloquent 模型中直接使用 hasIn() 方法了,它的用法与 has() 几乎完全一致,非常平滑:

use App\Models\User;

// 原始的 has() 方法,可能导致性能问题
// $users = User::has('posts')->paginate(10);
/*
  SQL:
  select * from `users`
  where exists
    (
       select * from `posts`
       where `users`.`id` = `posts`.`user_id`
    )
  limit 10 offset 0
*/

// 使用 hasIn() 替代,性能显著提升
$users = User::hasIn('posts')->paginate(10);
/*
  SQL:
  select * from `users`
  where `users`.`id` in
    (
       select `posts`.`user_id` from `posts`
    )
  limit 10 offset 0
*/

更多实用功能

红墨
红墨

一站式小红书图文生成器

下载

biiiiiiigmonster/hasin 不仅仅提供了 hasIn(),它还完美支持 Laravel has() 系列的所有变体,包括:

  • hasIn(): 基础的 WHERE IN 关系查询。

  • orHasIn(): OR 条件下的 hasIn

  • doesntHaveIn(): 查询不包含关联模型的记录。

  • orDoesntHaveIn(): OR 条件下的 doesntHaveIn

  • whereHasIn(): 在 hasIn 的基础上增加关联模型的条件。

    User::whereHasIn('posts', function ($query) {
        $query->where('votes', '>', 10);
    })->get();
  • hasMorphIn(): 支持多态关联的 hasIn

  • 嵌套关系: 同样支持 User::hasIn('posts.comments') 这样的嵌套关联查询。

优势与实际应用效果

引入 biiiiiiigmonster/hasin 后,我在那个电商项目中的查询性能得到了立竿见影的改善。原本需要数秒的页面加载时间,现在缩短到了几十毫秒,数据库的负载也明显降低。

其主要优势体现在:

  1. 显著的性能提升: 在外部表数据量大、内部表关联键字段有索引的场景下,WHERE INWHERE EXISTS 拥有更好的查询性能。
  2. 无缝集成: hasIn 系列方法的调用方式与 Laravel 原生的 has 方法保持一致,学习成本几乎为零。
  3. 功能完备: 支持所有 haswhereHas 的变体,以及多态关联和嵌套关联,覆盖了绝大多数使用场景。
  4. 易于维护: 避免了手写复杂 SQL 的需要,保持了 Eloquent ORM 的优雅和可读性。

总结

biiiiiiigmonster/hasin 是一个非常实用的 Laravel Composer 包,它巧妙地解决了 has 方法在处理大规模数据时可能出现的性能瓶颈。通过灵活选择 has()(基于 WHERE EXISTS,适用于内部表大、外部表小的情况)或 hasIn()(基于 WHERE IN,适用于外部表大、内部表小的情况),我们可以根据实际的数据量和业务场景,为 Laravel 应用选择最合适的查询策略,从而显著提升应用的响应速度和用户体验。如果你也正在为 Laravel 关系查询的性能问题而烦恼,不妨尝试一下 biiiiiiigmonster/hasin,它很可能会成为你的下一个“救星”!

相关专题

更多
laravel组件介绍
laravel组件介绍

laravel 提供了丰富的组件,包括身份验证、模板引擎、缓存、命令行工具、数据库交互、对象关系映射器、事件处理、文件操作、电子邮件发送、队列管理和数据验证。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

313

2024.04.09

laravel中间件介绍
laravel中间件介绍

laravel 中间件分为五种类型:全局、路由、组、终止和自定。想了解更多laravel中间件的相关内容,可以阅读本专题下面的文章。

270

2024.04.09

laravel使用的设计模式有哪些
laravel使用的设计模式有哪些

laravel使用的设计模式有:1、单例模式;2、工厂方法模式;3、建造者模式;4、适配器模式;5、装饰器模式;6、策略模式;7、观察者模式。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

363

2024.04.09

thinkphp和laravel哪个简单
thinkphp和laravel哪个简单

对于初学者来说,laravel 的入门门槛较低,更易上手,原因包括:1. 更简单的安装和配置;2. 丰富的文档和社区支持;3. 简洁易懂的语法和 api;4. 平缓的学习曲线。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

363

2024.04.10

laravel入门教程
laravel入门教程

本专题整合了laravel入门教程,想了解更多详细内容,请阅读专题下面的文章。

80

2025.08.05

laravel实战教程
laravel实战教程

本专题整合了laravel实战教程,阅读专题下面的文章了解更多详细内容。

63

2025.08.05

laravel面试题
laravel面试题

本专题整合了laravel面试题相关内容,阅读专题下面的文章了解更多详细内容。

62

2025.08.05

composer是什么插件
composer是什么插件

Composer是一个PHP的依赖管理工具,它可以帮助开发者在PHP项目中管理和安装依赖的库文件。Composer通过一个中央化的存储库来管理所有的依赖库文件,这个存储库包含了各种可用的依赖库的信息和版本信息。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

147

2023.12.25

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
第二十四期_PHP8编程
第二十四期_PHP8编程

共86课时 | 3.4万人学习

成为PHP架构师-自制PHP框架
成为PHP架构师-自制PHP框架

共28课时 | 2.4万人学习

第二十三期_PHP编程
第二十三期_PHP编程

共93课时 | 6.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号