0

0

如何在 PostgreSQL 中正确查询 JSON 类型字段中的指定角色

霞舞

霞舞

发布时间:2026-01-14 17:38:01

|

149人浏览过

|

来源于php中文网

原创

如何在 PostgreSQL 中正确查询 JSON 类型字段中的指定角色

本文详解在 symfony + postgresql 环境下,如何安全、高效地查询 `json`/`jsonb` 类型的 `roles` 字段中是否包含特定角色(如 `"role_ivm_user"`),避免 `operator does not exist: json = unknown` 错误。

在 PostgreSQL 中,JSON 类型字段不支持直接使用 IN、= 或 LIKE 等标准比较操作符进行数组元素匹配——这是因为 roles 存储的是 JSON 文本(如 ["ROLE_IVM_USER"]),而非关系型数组。你原写的 DQL 条件:

->andWhere(sprintf('%s.roles IN (:roles)', $rootAlias))
->setParameter('roles', [User::ROLE_IVM_USER]);

会被 Doctrine 转译为类似 WHERE roles IN ('ROLE_IVM_USER') 的 SQL,而 PostgreSQL 无法对 json 类型执行 IN 操作,故抛出 SQLSTATE[42883] 错误。

✅ 正确做法是利用 PostgreSQL 原生 JSONB 操作符(推荐将列类型升级为 JSONB,性能更优、功能更强),并结合 Doctrine 原生查询(createNativeQuery)精准匹配:

✅ 推荐方案:使用 ?? 操作符(JSONB 键存在检查)

?? 是 PostgreSQL JSONB 的专用操作符,用于判断某个字符串键(或元素值,在 jsonb 数组上下文中)是否存在。注意:它要求左操作数为 jsonb 类型,因此需显式转换:

谱乐AI
谱乐AI

谱乐AI,集成 Suno、Udio 等顶尖AI音乐模型的一站式AI音乐生成平台。

下载
use Doctrine\ORM\Query\ResultSetMappingBuilder;

$rsm = new ResultSetMappingBuilder($this->getEntityManager(), ResultSetMappingBuilder::COLUMN_RENAMING_INCREMENT);
$rsm->addRootEntityFromClassMetadata('App\Entity\User', 'u');

$rawSql = <<getEntityManager()->createNativeQuery($rawSql, $rsm);
$query->setParameter('role', User::ROLE_IVM_USER); // 传入字符串,如 "ROLE_IVM_USER"
return $query->getResult();
⚠️ 注意事项:表名 "user" 使用双引号是因为 user 是 PostgreSQL 保留关键字;?? 仅适用于 jsonb,若列仍为 json 类型,请先执行迁移: ALTER TABLE "user" ALTER COLUMN roles TYPE JSONB USING roles::JSONB;若需匹配嵌套结构(如 { "roles": ["ROLE_IVM_USER"] }),应改用 @> 操作符: WHERE u.roles::jsonb @> :role_json并传参 json_encode(['ROLE_IVM_USER'])。

? 替代方案(DQL 兼容,无需原生 SQL)

若坚持使用 DQL(如需复用 QueryBuilder),可借助 Doctrine 扩展(如 doctrine/dbal-postgresql-json)或自定义 DQL 函数,但原生方案更轻量、可控性更高。

✅ 总结

方案 是否推荐 说明
IN 直接比较 JSON 字段 类型不匹配,必然报错
::jsonb ?? :role(本文方案) ✅✅✅ 简洁、高效、原生支持,强烈推荐
@> + json_encode() 适合复杂嵌套结构,稍多一层序列化开销

掌握 jsonb 类型与 ??、@>、#> 等操作符,是驾驭 PostgreSQL JSON 数据的关键一步。务必确保数据库列类型为 JSONB,并在查询中显式转换,即可彻底规避此类类型错误。

相关专题

更多
PHP Symfony框架
PHP Symfony框架

本专题专注于PHP主流框架Symfony的学习与应用,系统讲解路由与控制器、依赖注入、ORM数据操作、模板引擎、表单与验证、安全认证及API开发等核心内容。通过企业管理系统、内容管理平台与电商后台等实战案例,帮助学员全面掌握Symfony在企业级应用开发中的实践技能。

78

2025.09.11

数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

676

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

320

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

346

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1095

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

357

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

675

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

571

2024.04.29

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

36

2026.01.14

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.3万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

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

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