在jsonb字段PostgreSQL中进行深度搜索

时间:2017-07-09 19:27:48

标签: postgresql jsonb

我的数据样本如下所示:

{"city": "NY", 
"skills": [
{"soft_skills": "Analysis"},
{"soft_skills": "Procrastination"},
{"soft_skills": "Presentation"}
],
"areas_of_training": [
{"areas of training": "Visio"},
{"areas of training": "Office"}, 
{"areas of training": "Risk Assesment"}
]}

我想运行查询以查找使用soft_skills Analysis的用户,并可能运行另一个查找用户,其中的培训范围为VisioRisk Assesment

我的列类型为jsonb。如何在这些深层嵌套的对象上实现搜索查询?使用SELECT * FROM mydata WHERE content::json->>'city'='NY';

在城市工程的第一级查询

如何使用LIKE关键字或字符串匹配为深层嵌套值运行匹配?

2 个答案:

答案 0 :(得分:3)

使用json_array_elements()获取嵌套元素的值,例如:

select d.*
from mydata d,
json_array_elements(content->'skills')
where value->>'soft_skills' ilike '%analysis%';

select d.*
from mydata d,
json_array_elements(content->'areas_of_training')
where value->>'areas of training' ~* 'visio|office';

查询可能会产生重复的行,因此使用select distinct on (id)是合理的,其中id是主键。

请注意,函数json_array_elements()代价很高,您无法使用与Abelisto解决方案相反的索引。但是,如果您想要访问嵌套的json元素的值,则必须使用它。

答案 1 :(得分:2)

1)

SELECT * FROM mydata
WHERE content->'skills' @> '[{"soft_skills": "Analysis"}]';

2)

SELECT * FROM mydata
WHERE content->'areas_of_training' @> '[{"areas of training": "Visio"},{"areas of training": "Risk Assesment"}]';

About JSON(B) operators

PS:准备好进行极慢的查询。我强烈建议您考虑data normalization

LIKE

的更新

对于您的示例数据,可能是:

SELECT * FROM mydata
WHERE EXISTS (
  SELECT *
  FROM jsonb_array_elements(content->'areas_of_training') as a
  WHERE a->>'areas of training' ilike '%vi%');

但是根据实际的JSON结构进行高度查询。