在PostgreSQL中,jsonb有额外的操作符,如 @>、<@、?、?|、?& 可以用来查询是否包含路径/值,以及顶层键值是否存在。
详细文章:PostgreSQL 操作json/jsonb
那么,如果我们不知道路径,只想要查询json/jsonb是否存在某个片段,那就跟我一起来学习吧!
思路:使用 ::text
将json/jsonb转换成为文本,再通过 like
关键字进行模糊查询。
示例:
select '{"a": {"b":{"c":"foo"}}}'::json::text like '%"c":"foo"%'; -- true
可以看到,结果返回true,这种思路是正确的。
不过。。。
再来看一组示例:
select '{"a": {"b":{"c":"foo"}}}'::json::text like '%"b":{"c": "foo"}%'; -- false
结果返回false,为什么呢?请思考一下。
细心的同学可能发现了,'%"b":{"c": "foo"}%'
这段文本的 "c":
后面多了一个空格。
那好吧,把冒号后面的空格删除掉,示例:
select '{"a": {"b":{"c":"foo"}}}'::jsonb::text like '%"c":"foo"%'; -- false
结果还是返回false,为什么呢?
冒号后面加上空格
select '{"a": {"b":{"c":"foo"}}}'::jsonb::text like '%"b": {"c": "foo"}%'; -- true
结果返回true,为什么呢?两段文本明明就不匹配。
因为我偷偷把 json
换成了 jsonb
。
看看 json
和 josnb
转换成text的格式。
select '{"a": {"b":{"c":"foo"}}}'::jsonb; -- {"a": {"b": {"c": "foo"}}}
select '{"a": {"b":{"c":"foo"}}}'::json; -- {"a": {"b":{"c":"foo"}}}
有没有恍然大悟?