0

0

Laravel如何使用Eloquent进行子查询

尼克

尼克

发布时间:2025-12-01 10:17:06

|

970人浏览过

|

来源于php中文网

原创

Laravel Eloquent支持子查询,可通过addSelect获取用户最新订单时间,where条件中嵌套查询筛选订单总额超1000的用户,joinSub将子查询结果作为临时表连接,或使用withSum等快捷方法预加载聚合数据,提升复杂查询可读性与效率。

laravel如何使用eloquent进行子查询

在 Laravel 中,Eloquent 支持通过子查询(Subquery)来实现更复杂的查询逻辑。你可以将一个查询的结果作为另一个查询的字段或条件使用,尤其适用于获取关联数据中的最新记录、统计值等场景。下面介绍几种常见的使用方式。

在 SELECT 中使用子查询

如果你想在查询结果中包含一个来自子查询的字段,比如获取每个用户的最后一条订单信息,可以这样做:

假设你有 User 模型和 Order 模型,你想查出每个用户的名字和他们最近订单的创建时间。

use Illuminate\Database\Eloquent\Builder;

$users = User::select('users.*')
    ->addSelect(['last_order_at' => Order::select('created_at')
        ->whereColumn('user_id', 'users.id')
        ->orderByDesc('created_at')
        ->limit(1)
    ])
    ->get();

这里使用了 addSelect() 添加一个子查询字段,该子查询获取当前用户最新的订单时间。

在 WHERE 条件中使用子查询

你可以用子查询作为 WHERE 的判断条件。例如:找出订单总额超过 1000 的用户。

$totalQuery = Order::selectRaw('sum(amount)')
    ->whereColumn('user_id', 'users.id');

$users = User::whereHas('orders')
    ->where(function (Builder $query) use ($totalQuery) {
        $query->select($totalQuery)->from('orders')->limit(1)->shouldAllowSelfJoins()
    }, '>', 1000)
    ->get();

也可以更简洁地写成:

ASP.NET 4.0电子商城
ASP.NET 4.0电子商城

在现实生活中的购物过程,购物者需要先到商场,找到指定的产品柜台下,查看产品实体以及标价信息,如果产品合适,就将该产品放到购物车中,到收款处付款结算。电子商务网站通过虚拟网页的形式在计算机上摸拟了整个过程,首先电子商务设计人员将产品信息分类显示在网页上,用户查看网页上的产品信息,当用户看到了中意的产品后,可以将该产品添加到购物车,最后使用网上支付工具进行结算,而货物将由公司通过快递等方式发送给购物者

下载
$users = User::whereExists(function ($query) {
    $query->select(DB::raw(1))
          ->from('orders')
          ->whereColumn('orders.user_id', 'users.id')
          ->havingRaw('sum(amount) > 1000')
          ->groupBy('orders.user_id');
})->get();

在 JOIN 中使用子查询

有时你需要将子查询作为一个临时表进行 JOIN。例如获取每个用户的最新一条订单。

$latestOrderSubquery = Order::select('user_id', DB::raw('max(created_at) as last_order_at'))
    ->groupBy('user_id');

$users = User::joinSub($latestOrderSubquery, 'latest_orders', function ($join) {
    $join->on('users.id', '=', 'latest_orders.user_id');
})->get();

joinSub() 方法允许你把一个查询当作一个子查询表来 JOIN,非常适用于聚合或去重场景。

使用 withSum、withMax 等快捷方法(Laravel 8+)

Laravel 提供了一些便捷的预加载子查询方法,如 withSumwithMaxwithAvg 等。

// 获取用户及其订单总金额
$users = User::withSum('orders as total_spent', 'amount')->get();

foreach ($users as $user) {
    echo $user->total_spent; // 直接访问
}

这种方式比手动写子查询更清晰,适合简单的聚合场景。

基本上就这些。Laravel 的 Eloquent 子查询功能强大且灵活,合理使用能让复杂查询变得简洁易懂。关键在于理解何时使用 selectSubjoinSub 或内置的 withXxx 方法。不复杂但容易忽略细节,比如别名定义和作用域绑定。

相关专题

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

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

316

2024.04.09

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

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

275

2024.04.09

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

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

369

2024.04.09

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

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

371

2024.04.10

laravel入门教程
laravel入门教程

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

81

2025.08.05

laravel实战教程
laravel实战教程

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

64

2025.08.05

laravel面试题
laravel面试题

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

67

2025.08.05

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

11

2026.01.19

微信聊天记录删除恢复导出教程汇总
微信聊天记录删除恢复导出教程汇总

本专题整合了微信聊天记录相关教程大全,阅读专题下面的文章了解更多详细内容。

73

2026.01.18

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Laravel---API接口
Laravel---API接口

共7课时 | 0.6万人学习

PHP自制框架
PHP自制框架

共8课时 | 0.6万人学习

PHP面向对象基础课程(更新中)
PHP面向对象基础课程(更新中)

共12课时 | 0.7万人学习

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

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