在 Laravel Eloquent 中高效查询 JSON 数组字段的教程

在 Laravel Eloquent 中高效查询 JSON 数组字段的教程

本文旨在解决 laravel eloquent 在查询 json 数组字段时遇到的路径解析问题。当需要根据 json 数组中特定索引的值进行筛选时,eloquent 的 `wherejsondoesntcontain` 或 `where` 方法可能因生成错误的 sql 路径(例如 `$.”key”` 而非 `$[index]`)而失效。文章将详细解释此问题,并提供使用 `whereraw` 结合 `json_extract` 的有效解决方案,同时指出 laravel 9.0 及更高版本对此的改进。

Laravel Eloquent 查询 JSON 数组字段:问题与解决方案

在 Laravel 应用开发中,我们经常会遇到需要存储和查询 JSON 类型数据的情况。Eloquent ORM 提供了便捷的方法来处理 JSON 字段,例如 whereJsonDoesntContain 或使用 -> 运算符直接访问 JSON 路径。然而,当 JSON 字段存储的是数组,并且我们需要根据数组的特定索引值进行筛选时,可能会遇到意料之外的行为。

问题描述:Eloquent 对 JSON 数组路径的误解

考虑一个 Book 模型,其中包含一个 readings 字段,该字段是一个 JSON 数组,记录了每个月书籍的阅读量,例如 [1, 0, 3, 2, 5, 5, 2, 1, 3, 0, 0, 2]。我们希望查询每个月中阅读量不为零(即有阅读量)的书籍。

直观上,我们可能会尝试使用 whereJsonDoesntContain 方法,如下所示:

for ($i = 1; $i '. ($i - 1), 0)->get();    // ...}

或者使用更通用的 where 方法:

for ($i = 1; $i '. ($i - 1), '!=', 0)->get();    // ...}

然而,这些查询在实际执行时可能无法返回预期的结果。原因是 Laravel 在内部将这些 Eloquent 调用转换为 SQL 查询时,对于 JSON 数组的索引路径处理存在偏差。

例如,Book::whereJsonDoesntContain(‘readings->3’, 0) 会被转换为类似于以下 SQL 语句:

SELECT * FROM books WHERE NOT JSON_CONTAINS(`readings`, '0', '$."3"')

而 Book::where(‘readings->3’, ‘!=’, 0) 则会被转换为:

SELECT * FROM books WHERE JSON_UNQUOTE(JSON_EXTRACT(`readings`, '$."3"')) != '0'

问题在于 SQL 的 JSON 函数(如 JSON_CONTAINS 或 JSON_EXTRACT)在访问数组元素时,需要使用 $[index] 这样的路径格式,而不是 $.”key”。Laravel 默认将 -> 后面的数字视为对象键,导致生成的 SQL 路径不正确,从而无法准确查询 JSON 数组中的特定索引。

解决方案:使用 whereRaw 和 JSON_EXTRACT

为了正确地查询 JSON 数组的特定索引,我们需要绕过 Eloquent 自动生成的路径,直接使用 whereRaw 方法来构造 SQL 语句。通过 JSON_EXTRACT 函数,我们可以精确地指定数组元素的路径 $[index]。

以下是修正后的查询方法:

for ($i = 0; $i get();    // ...}

代码解释:

for ($i = 0; $i Book::whereRaw(…): 允许我们直接编写原始 SQL 条件。JSON_EXTRACT(readings`, ‘$[$i]’)`: 这是关键部分。readings: 是你的 JSON 字段名。’$[$i]’: 这是正确的 JSON 路径表达式。$: 在 PHP 字符串中,$ 需要转义,以确保它被视为 SQL 路径的一部分,而不是 PHP 变量。[$i]: 表示访问 JSON 数组中索引为 $i 的元素。!= 0: 筛选出该月阅读量不为零的记录。

通过这种方式,生成的 SQL 语句将是正确的,能够准确地从 JSON 数组中提取指定索引的值进行比较。

注意事项与版本兼容性

数据库兼容性:JSON_EXTRACT 和 JSON_CONTAINS 等 JSON 函数是 MySQL 5.7+、PostgreSQL 9.4+ 等现代关系型数据库支持的特性。请确保你的数据库版本支持这些函数。

性能考量:对 JSON 字段进行查询通常会比查询普通列的性能开销大。如果查询频率高且数据量大,可以考虑将关键数据点提取为单独的列,或者为 JSON 字段创建合适的索引(例如,MySQL 8.0+ 支持 JSON 路径索引)。

Laravel 9.0+ 的改进:值得注意的是,Laravel 9.0 及更高版本对 JSON 数组路径的处理进行了改进。从 Laravel 9.0 开始,你可以通过以下语法正确地访问 JSON 数组元素:

// 在 Laravel 9.0+ 中,这将正确生成 $[3] 这样的路径Book::whereJsonDoesntContain('readings->[3]', 0);

这意味着在较新版本的 Laravel 中,你可能不再需要 whereRaw 来处理这种特定场景。然而,了解 whereRaw 的用法仍然是处理复杂或特定数据库函数查询时的重要技能。

总结

在 Laravel Eloquent 中查询 JSON 数组字段的特定索引时,由于 Eloquent 早期版本对 JSON 路径解析的限制,直接使用 whereJsonDoesntContain 或 where 可能会生成错误的 SQL。通过 whereRaw 结合 JSON_EXTRACT 函数,并使用正确的 ‘$[index]’ 路径语法,可以有效地解决这个问题。对于使用 Laravel 9.0 及更高版本的开发者,框架本身已经对此进行了优化,提供了更简洁的语法支持。在选择解决方案时,应根据你的 Laravel 版本和数据库兼容性进行权衡。

以上就是在 Laravel Eloquent 中高效查询 JSON 数组字段的教程的详细内容,更多请关注php中文网其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1334285.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月12日 20:10:15
下一篇 2025年12月12日 20:10:32

相关推荐

  • 怎么获取php源码_php获取源码安全渠道与授权注意【指南】

    首先从官方GitHub仓库克隆PHP源码以确保安全性和实时性,具体操作为访问https://github.com/php/php-src并使用git clone命令下载,随后可通过git checkout切换至所需版本分支。其次可从PHP官网下载经GPG签名验证的正式发布包,确保文件完整性,需在终端…

    2025年12月13日
    000
  • WordPress中强制设置文章标题及页面文本方向为左到右的教程

    本教程旨在指导WordPress用户如何调整网站内容(特别是文章标题)的文本方向,使其从默认或自动识别的右到左(RTL)方向强制显示为左到右(LTR)。我们将通过修改主题文件来实现全局LRT设置,并探讨更具针对性的解决方案,以适应多语言网站的特定需求,确保内容显示符合预期。 理解网页文本方向与Wor…

    2025年12月13日
    000
  • PHP中CI框架的运行模式

    CodeIgniter的“运行模式”并非严格定义,而是通过配置和代码控制的执行流程风格:1. 单入口模式为默认方式,所有请求经index.php进入,由CodeIgniter.php驱动完整生命周期;2. CLI模式支持命令行执行,用于定时任务等场景,通过is_cli()识别环境并跳过Web专属逻辑…

    2025年12月13日
    000
  • php源码怎么进后台_php源码进入后台与权限设置【方法】

    首先确认后台入口路径,检查常见目录如admin或登录文件;接着通过数据库查看或修改管理员账号密码;再检查配置文件开启调试模式或添加会话绕过登录;然后设置正确文件权限确保可读写;最后尝试URL参数触发隐藏激活机制。 如果您拥有PHP源码项目并希望进入其后台管理系统,但不清楚如何访问或配置权限,则可能是…

    2025年12月13日
    000
  • php怎么通过md5解密出来_用PHP逆向md5加密获取明文教程【技巧】

    MD5不可逆,无法直接还原,但可通过彩虹表查询、本地字典比对或在线API批量查询尝试匹配明文。 如果您尝试对一个经过MD5处理的字符串进行还原,需要明确的是MD5是一种单向哈希算法,设计目的就是不可逆。因此无法通过传统意义上的“解密”来直接还原原始数据。但可以通过查找已知明文与对应哈希值的方式尝试匹…

    2025年12月13日
    000
  • php源码怎么安装_用PHP环境安装源码步骤教程【教程】

    首先检查并搭建PHP运行环境,安装XAMPP等集成环境,启动Apache和MySQL服务,将源码放入htdocs或www目录,通过localhost访问项目;接着配置数据库连接信息,修改config.php等文件中的数据库参数,创建数据库并导入SQL文件;然后设置文件权限,确保uploads、cac…

    2025年12月13日
    000
  • 找php源码怎么破解_找php源码破解风险与合法建议【警示】

    使用PHP源码破解工具违法且危险,一、非法破解侵犯知识产权并可能植入后门,需检查来源、分析可疑函数、扫描病毒;二、避免第三方解密工具,拒传源码、禁用未知脚本、排查新文件;三、应通过开源平台、购买授权或自研代码合法获取;四、加强防护,关闭危险函数、限制目录权限、启用Open_basedir。 如果您在…

    2025年12月13日
    000
  • ZKTeco考勤数据集成至Google Sheets或在线服务器实战教程

    本教程详细介绍了如何将ZKTeco考勤机(如K40、F18)的考勤数据集成到Google Sheets或自定义在线服务器。通过开发中间程序获取设备数据,并在服务器端进行处理,最终利用Google Apps Script的UrlFetch服务将JSON数据导入Google Sheets,有效解决了考勤…

    2025年12月13日
    000
  • 怎么偷php网站源码_偷php网站源码违法与防护建议【警示】

    可通过开源平台、官方渠道或开发工具合法获取PHP源码。1、从GitHub等平台克隆开源项目;2、在php.net下载PHP解释器源码;3、用浏览器开发者工具查看前端代码;4、本地部署WordPress等开源应用学习;5、通过配置服务器、隔离配置文件、更新系统加强防护。 如果您发现某个网站可能存在安全…

    2025年12月13日
    000
  • php数组排序并输出

    PHP中排序数组需根据类型选择函数:1. sort()对索引数组按值升序排序;2. ksort()按键排序关联数组;3. rsort()和arsort()分别对索引和关联数组按值降序排序;4. usort()配合自定义比较函数实现灵活排序,如按学生分数降序排列;所有sort系列函数均修改原数组,输出…

    2025年12月13日
    000
  • 解决CodeIgniter 4中表单提交后重定向失败的CSP问题

    当CodeIgniter 4应用中的表单提交后重定向功能失效,尤其是在Chrome浏览器中出现“Refused to send form data… form-action ‘self’”的错误时,这通常是由于内容安全策略(CSP)中的`form-action &…

    2025年12月13日
    000
  • PHP路由中通过call_user_func_array传递参数的技巧与实践

    本文深入探讨了在php自定义路由中,如何利用正则表达式从url中提取动态参数,并通过`call_user_func_array`机制将这些参数灵活地传递给控制器方法。文章通过构建一个简化的路由系统,详细阐述了路由匹配、参数捕获以及方法调用的全过程,并提供了实用的代码示例和注意事项,旨在帮助开发者理解…

    2025年12月13日
    000
  • 在PHP中安全处理并发送多选框(Checkbox)值到邮件

    本教程详细阐述了如何在PHP中正确处理HTML表单提交的多选框(Checkbox)数组值,并将其格式化以便通过邮件发送。文章涵盖了从HTML结构到PHP数组处理的关键步骤,包括使用`implode`函数将数组转换为字符串,以及至关重要的安全实践,如使用`htmlentities`或`htmlspec…

    2025年12月13日
    000
  • 使用Ajax从超链接动态传递GET参数到PHP页面

    本教程详细讲解如何利用jquery ajax技术,从html超链接的`href`属性中动态获取get参数,并将其发送至php后端进行处理,从而实现页面无刷新数据交互。文章将涵盖从前端javascript拦截链接点击、构建ajax请求到后端php接收数据的完整流程,并提供示例代码和注意事项。 在Web…

    2025年12月13日 好文分享
    000
  • 使用 Guzzle HTTP 和 Goutte 模拟表单登录教程

    直接使用 Guzzle 的 `auth` 选项通常不适用于模拟基于表单的网站登录。本教程将指导您如何通过模拟浏览器行为实现表单登录,包括首先通过 GET 请求获取登录页面以提取表单数据(如 CSRF 令牌),然后使用 POST 请求提交凭据,并确保会话管理以进行后续的认证请求。 在开发涉及与外部网站…

    2025年12月13日
    000
  • SQL与PHP实现课程学生并发量精确统计教程

    本教程详细阐述了在mysql 5.6和php 7.2环境下,如何精确统计指定课程在特定日期范围内的学生并发量。针对传统查询无法准确处理日期区间重叠的问题,文章提出并演示了利用“日历表”结合sql聚合函数,有效计算每日活跃学生数,并从中找出指定时间段内的最大并发峰值,确保统计结果的准确性与可靠性。 挑…

    2025年12月13日
    000
  • PHP密码长度验证:常见陷阱与最佳实践

    本教程详细探讨php中密码长度验证的正确实现方法,重点纠正常见的逻辑错误,并强调使用`mb_strlen`处理多字节字符的重要性。文章将指导开发者如何构建健壮且易读的验证函数,并将其无缝集成到表单提交流程中,同时提供代码示例和优化建议,确保密码验证的准确性和安全性。 密码长度验证的重要性与常见误区 …

    2025年12月13日
    000
  • Nginx自定义错误页面:实现外部跳转与邮件通知

    本文详细阐述如何在nginx中配置自定义错误页面,以实现当服务器出现4xx或5xx错误时,不仅能将用户重定向到指定的外部网站,还能同时触发后端php脚本发送邮件通知。教程涵盖nginx配置、php脚本实现及curl命令行测试方法,旨在提供一个灵活且功能强大的错误处理机制。 在网站运维中,优雅地处理服…

    2025年12月13日
    000
  • 利用PHP DateTime类处理日期输入与月份识别

    本文旨在解决%ignore_a_1%中根据用户提交日期判断其所属月份的问题。传统的多条件if-else或switch-case语句在处理日期范围时效率低下且易出错。本教程将介绍如何利用php内置的datetime类,简洁高效地解析日期字符串,并直接提取或格式化出月份信息,从而避免复杂的日期区间比较,…

    2025年12月13日
    000
  • PHP PDO实现用户密码条件更新:当输入为空时不修改密码

    本教程详细阐述了在php pdo应用中,如何实现用户密码的条件式更新。当用户在更新表单中未输入新密码时,系统将保留数据库中已有的密码,避免不必要的修改。文章通过优化sql的`if`语句,结合php的输入处理和安全实践,提供了一个健壮且高效的解决方案,同时纠正了常见的语法错误,确保数据更新的灵活性和准…

    2025年12月13日
    000

发表回复

登录后才能评论
关注微信