将原生子查询转换为 Laravel Query Builder 查询

霞舞
发布: 2025-10-03 16:44:01
原创
657人浏览过

将原生子查询转换为 laravel query builder 查询

本文旨在指导开发者如何使用 Laravel 的 Query Builder 将包含子查询的原生 SQL 查询转换为 Laravel 风格的查询。通过 DB::select 和 fromSub 方法,我们将演示如何构建嵌套查询,并处理 whereIn 等复杂条件,从而提高代码的可读性和可维护性。本文将提供详细的代码示例,帮助您理解和应用这一技术。

在 Laravel 开发中,我们经常需要执行复杂的 SQL 查询。虽然可以使用原生 SQL 语句,但 Laravel 的 Query Builder 提供了更安全、更易于维护的方式来构建查询。当涉及到包含子查询的复杂查询时,Query Builder 同样能够胜任。本文将介绍如何将原生 SQL 子查询转换为 Laravel Query Builder 查询。

使用 fromSub 构建子查询

Laravel 提供了 fromSub 方法来处理子查询。该方法接受两个参数:一个闭包函数,用于定义子查询;以及一个别名,用于引用子查询的结果集。

让我们考虑以下原生 SQL 查询:

SELECT inventory.EmployeeID,
        inventory.created_date AS OrderDate,
        SUM(inventory.calculation) AS TotalPrice
        FROM ( SELECT i.id AS ItemID,
        o.id AS OrderID,
        o.EmployeeID,
        o.created_date,
        (o.Quantity * i.price) AS calculation
        FROM `stationary_orders` AS o
        LEFT JOIN `stationary_items` AS i ON o.Stationary_ID = i.id
        WHERE o.Store IN $storess
        ORDER BY o.id DESC
        LIMIT $Limit,10 ) AS inventory
        GROUP BY inventory.EmployeeID
登录后复制

要将其转换为 Laravel Query Builder 查询,可以使用以下代码:

use Illuminate\Support\Facades\DB;

$stores = ['store1', 'store2', 'store3']; // 示例数据
$limit = 10; // 示例数据

$result = DB::table(DB::raw('( SELECT i.id AS ItemID,
        o.id AS OrderID,
        o.EmployeeID,
        o.created_date,
        (o.Quantity * i.price) AS calculation
        FROM `stationary_orders` AS o
        LEFT JOIN `stationary_items` AS i ON o.Stationary_ID = i.id
        WHERE o.Store IN ("'.implode('","', $stores).'")
        ORDER BY o.id DESC
        LIMIT '.$limit.',10 ) AS inventory'))
    ->select('inventory.EmployeeID', 'inventory.created_date AS OrderDate', DB::raw('SUM(inventory.calculation) AS TotalPrice'))
    ->groupBy('inventory.EmployeeID')
    ->get();

// 或者使用 fromSub 方法,更加安全
$result = DB::table(DB::raw('( SELECT i.id AS ItemID,
        o.id AS OrderID,
        o.EmployeeID,
        o.created_date,
        (o.Quantity * i.price) AS calculation
        FROM `stationary_orders` AS o
        LEFT JOIN `stationary_items` AS i ON o.Stationary_ID = i.id
        WHERE o.Store IN ("'.implode('","', $stores).'")
        ORDER BY o.id DESC
        LIMIT '.$limit.',10 ) AS inventory'))
    ->select('inventory.EmployeeID', 'inventory.created_date AS OrderDate', DB::raw('SUM(inventory.calculation) AS TotalPrice'))
    ->groupBy('inventory.EmployeeID')
    ->get();
登录后复制

或者使用更加安全的fromSub方法

use Illuminate\Support\Facades\DB;

$stores = ['store1', 'store2', 'store3']; // 示例数据
$limit = 10; // 示例数据

$result = DB::query()
    ->select(DB::raw('inventory.EmployeeID, inventory.created_date AS OrderDate, SUM(inventory.calculation) AS TotalPrice'))
    ->fromSub(function ($query) use ($stores, $limit) {
        $query->select(DB::raw('i.id AS ItemID,
        o.id AS OrderID,
        o.EmployeeID,
        o.created_date,
        (o.Quantity * i.price) AS calculation'))
            ->from('stationary_orders AS o')
            ->leftJoin('stationary_items AS i', 'o.Stationary_ID', '=', 'i.id')
            ->whereIn('o.Store', $stores)
            ->orderBy('o.id', 'desc')
            ->limit(10)
            ->offset($limit);
    }, 'inventory')
    ->groupBy('inventory.EmployeeID')
    ->get();
登录后复制

代码解释:

  1. DB::query(): 创建一个新的数据库查询构建器实例。

    百度智能云·曦灵
    百度智能云·曦灵

    百度旗下的AI数字人平台

    百度智能云·曦灵 83
    查看详情 百度智能云·曦灵
  2. select(DB::raw(...)): 选择需要返回的字段,使用了 DB::raw() 来执行原生 SQL 函数,例如 SUM()。

  3. fromSub(function ($query) use ($stores, $limit) { ... }, 'inventory'): 定义子查询。

    • function ($query) use ($stores, $limit): 一个闭包函数,接收一个 $query 对象,用于构建子查询。 use ($stores, $limit) 将外部变量 $stores 和 $limit 传递到闭包内部。
    • $query->select(...): 在子查询中选择需要的字段。
    • $query->from('stationary_orders AS o'): 指定子查询的表。
    • $query->leftJoin(...): 执行左连接。
    • $query->whereIn('o.Store', $stores): 使用 whereIn() 方法处理 WHERE IN 条件。
    • $query->orderBy('o.id', 'desc'): 对结果进行排序。
    • $query->limit(10): 限制结果数量。
    • $query->offset($limit): 设置偏移量。
    • 'inventory': 为子查询的结果集指定别名 inventory。
  4. groupBy('inventory.EmployeeID'): 根据 inventory.EmployeeID 进行分组。

  5. get(): 执行查询并返回结果。

使用 whereIn 处理数组条件

在上面的示例中,WHERE o.Store IN $storess 条件被转换为了 $query->whereIn('o.Store', $stores)。whereIn 方法接受两个参数:要比较的字段名和一个包含值的数组。这使得处理 IN 条件变得非常简单。

注意事项

  • SQL 注入风险: 使用原生 SQL 语句时,务必注意 SQL 注入风险。使用 Query Builder 可以有效避免这种风险,因为它会自动对参数进行转义。
  • 性能: 复杂的子查询可能会影响性能。在实际应用中,应该仔细评估查询的性能,并考虑使用索引等优化手段。
  • 可读性: 尽管 Query Builder 可以构建复杂的查询,但过度复杂的查询可能会降低代码的可读性。在必要时,可以考虑将复杂的查询拆分为多个较小的查询。
  • 变量传递: 使用 use 关键字将外部变量传递到闭包函数中,确保子查询可以访问必要的参数。

总结

通过 fromSub 方法,我们可以轻松地将原生 SQL 子查询转换为 Laravel Query Builder 查询。这不仅提高了代码的可读性和可维护性,还有助于避免 SQL 注入风险。在实际开发中,应根据具体情况选择合适的查询构建方式,并注意性能优化。

以上就是将原生子查询转换为 Laravel Query Builder 查询的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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