根据一定条件替换所有json数组元素中的键/值对
我有这样的表:
create table view (id int, content jsonb);
insert into view values (0,'[{"owner":"foo", "value": 1},{"owner":"bar", "value": 2},{"owner":"foo", "value": 3}]'::jsonb);
insert into view values (1,'[{"owner":"bar", "value": 3},{"owner":"bar", "value": 5},{"owner":"foo", "value": 6}]'::jsonb);
insert into view values (2,'[{"owner":"foo", "value": 4},{"owner":"foo", "value": 8},{"owner":"bar", "value": 9}]'::jsonb);
如果原始值为“foo”,我想用值“foo1”重新填充每个数组元素中“owner”的值,其他所有内容都应保持不变。
我最终得到这样的更新查询:
WITH content_item AS (
SELECT
('{' || index - 1 || ',"owner"}')::TEXT[] AS path,
id,
(item -> 'owner') as owner
FROM view,
jsonb_array_elements(content) WITH ORDINALITY arr(item, index)
)
UPDATE view v
SET content = jsonb_set(content, content_item.path, '"foo1"')
FROM content_item
WHERE owner = '"foo"' AND content_item.id = v.id;
问题是只有第一次出现的“foo”被替换为“foo1”:
id |content
------------------------------------------------------------------------------------------------
0 |[{"owner": "foo1", "value": 1}, {"owner": "bar", "value": 2}, {"owner": "foo", "value": 3}]
1 |[{"owner": "bar", "value": 3}, {"owner": "bar", "value": 5}, {"owner": "foo1", "value": 6}]
2 |[{"owner": "foo1", "value": 4}, {"owner": "foo", "value": 8}, {"owner": "bar", "value": 9}]
如何替换数组中的所有出现?
I have table like this:
create table view (id int, content jsonb);
insert into view values (0,'[{"owner":"foo", "value": 1},{"owner":"bar", "value": 2},{"owner":"foo", "value": 3}]'::jsonb);
insert into view values (1,'[{"owner":"bar", "value": 3},{"owner":"bar", "value": 5},{"owner":"foo", "value": 6}]'::jsonb);
insert into view values (2,'[{"owner":"foo", "value": 4},{"owner":"foo", "value": 8},{"owner":"bar", "value": 9}]'::jsonb);
I would like to replate the value of "owner" in every array element with value "foo1" if an orignal value is "foo", everything else should stay intact.
I end up with update query like this:
WITH content_item AS (
SELECT
('{' || index - 1 || ',"owner"}')::TEXT[] AS path,
id,
(item -> 'owner') as owner
FROM view,
jsonb_array_elements(content) WITH ORDINALITY arr(item, index)
)
UPDATE view v
SET content = jsonb_set(content, content_item.path, '"foo1"')
FROM content_item
WHERE owner = '"foo"' AND content_item.id = v.id;
Problem is that only a first occurrence of "foo" is replaced with "foo1":
id |content
------------------------------------------------------------------------------------------------
0 |[{"owner": "foo1", "value": 1}, {"owner": "bar", "value": 2}, {"owner": "foo", "value": 3}]
1 |[{"owner": "bar", "value": 3}, {"owner": "bar", "value": 5}, {"owner": "foo1", "value": 6}]
2 |[{"owner": "foo1", "value": 4}, {"owner": "foo", "value": 8}, {"owner": "bar", "value": 9}]
How do i replace all occurrences within an array?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
data:image/s3,"s3://crabby-images/d5906/d59060df4059a6cc364216c4d63ceec29ef7fe66" alt="扫码二维码加入Web技术交流群"
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
使用基于文本的方法:
请参阅现场演示。
注意:Postgres 不支持从文本直接转换为 jsonb,但它会将文本转换为 json,然后在更新期间自动转换为 jsonb。
还有一个风格问题:如果可能,最好不要在实体名称中使用保留字,例如
view
。Use a text-based approach:
See live demo.
Note: Postgres doesn’t support casting from text straight to jsonb, but it will cast text to json then auto-cast to jsonb during the update.
Also a style thing: If possible, it’s best to not use reserved words, eg
view
, for entity names.一旦将 UPDATE 转换为 SELECT,问题就变得显而易见 - 对于每次出现的owner = 'foo',您都会得到一行,然后替换该出现的行,但另一行保持不变。
当对每一行执行 UPDATE 时,它将更新出现 n 次的行 n 次,每次保持 n-1 次出现完好。
可能有更好的方法,但您可以做的是递归地合并所有这些更新,一个接一个,然后最后您将获得最终版本,并使用该版本进行更新。
因此,在第一个 CTE 中,我使用 ROW_NUMBER() 为每个出现的“foo”生成一个索引,作为所有者。这为我们提供了所有需要更改的事件的详细分类。然后,在递归 CTE 中,我从索引 = 1(第一次出现“foo”)开始逐一应用每个更改。最终,我使用最后一个(相关子查询来获取每个 id 的 MAX(索引))来执行 UPDATE。
我相当确信应该有一种更简单、更优雅的方法来做到这一点,并且会很高兴地对提供的答案进行投票。也许您会发现我的回答同时有用。
这是 dbfiddle 的工作演示。
Once you convert your UPDATE into a SELECT, the issue becomes obvious - for each occurrence of owner = 'foo' you get a row, and then you replace that occurrence, but the other one remains the same.
As UPDATE executes for each of the rows, it will update the rows with n occurrences n times, each time leaving n-1 occurrences intact.
There might be a better way, but what you can do is merge all these updates recursively, one after another, and then at the end you'll get the final version, and you use that version for the update.
So in the first CTE the addition is I generate an index for each of the occurrences of "foo" as the owner using
ROW_NUMBER()
. This gives us a nice little breakdown of all the occurrences that we need to change. Then later in the recursively CTE, I apply each of the changes one by one, starting from index = 1 (first occurrence of "foo"). Eventually at the end I use the last (correlated subquery to get MAX(index) for each id) to execute the UPDATE.I am fairly convinced that there should be a simpler and more elegant way to do it and will happily upvote the answer that provides one. Maybe you'll find my answer useful in the meantime.
Here's a working demo on dbfiddle.