如何修改新PostgreSQLJSON數據類型中的字段?


以下內容轉自:

https://cloud.tencent.com/developer/ask/50349

如何修改新PostgreSQLJSON數據類型中的字段?

 
  • 回答 (2)
  • 關注 (0)
  • 查看 (341)

使用PostgreSQL 9.3,我可以選擇JSON數據類型的特定字段,但是如何使用UPDATE修改它們呢?我在PostgreSQL文檔中或在線的任何地方都找不到這方面的任何例子。我嘗試過顯而易見的:

postgres=# create table test (data json); CREATE TABLE postgres=# insert into test (data) values ('{"a":1,"b":2}'); INSERT 0 1 postgres=# select data->'a' from test where data->>'b' = '2'; ?column? ---------- 1 (1 row) postgres=# update test set data->'a' = to_json(5) where data->>'b' = '2'; ERROR: syntax error at or near "->" LINE 1: update test set data->'a' = to_json(5) where data->>'b' = '2...
提問於 2018-02-262018-02-26 08:00:20

如何修改新PostgreSQLJSON數據類型中的字段?

2 個回答

用戶回答回答於 2018-02-262018-02-26 16:33:55

合並2個(或更多)JSON對象(或連接數組):

SELECT jsonb '{"a":1}' || jsonb '{"b":2}', -- will yield jsonb '{"a":1,"b":2}' jsonb '["a",1]' || jsonb '["b",2]' -- will yield jsonb '["a",1,"b",2]'

所以,設置一個簡單的鍵可以使用:

SELECT jsonb '{"a":1}' || jsonb_build_object('<key>', '<value>')

何地<key>應該是字符串,而且<value>可以是任何類型的to_jsonb()接受。

為在JSON層次結構中深層設置一個值,jsonb_set()功能可用於:

SELECT jsonb_set('{"a":[null,{"b":[]}]}', '{a,1,b,0}', jsonb '{"c":3}') -- will yield jsonb '{"a":[null,{"b":[{"c":3}]}]}'

全參數列表jsonb_set():

jsonb_set(target jsonb, path text[], new_value jsonb, create_missing boolean default true)

path也可以包含JSON數組索引&出現在那里的負整數從JSON數組的末尾開始計數。但是,一個不存在但正的JSON數組索引將將元素追加到數組的末尾:

SELECT jsonb_set('{"a":[null,{"b":[1,2]}]}', '{a,1,b,1000}', jsonb '3', true) -- will yield jsonb '{"a":[null,{"b":[1,2,3]}]}'

為插入JSON數組(同時保留所有原始值),jsonb_insert()函數可以使用:

SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,b,0}', jsonb '2') -- will yield jsonb '{"a":[null,{"b":[2,1]}]}', and SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,b,0}', jsonb '2', true) -- will yield jsonb '{"a":[null,{"b":[1,2]}]}'

全參數列表jsonb_insert():

jsonb_insert(target jsonb, path text[], new_value jsonb, insert_after boolean default false)

中出現的.path從JSON數組的末尾計數。

因此,將f.exe.追加到JSON數組的末尾可以通過以下方式完成:

SELECT jsonb_insert('{"a":[null,{"b":[1,2]}]}', '{a,1,b,-1}', jsonb '3', true) -- will yield jsonb '{"a":[null,{"b":[1,2,3]}]}', and

但是,此功能的工作方式略有不同(與jsonb_set())當pathtarget是JSON對象的密鑰。在這種情況下,它只會在不使用密鑰時為JSON對象添加一個新的鍵值對。如果使用它,它將引發一個錯誤:

SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,c}', jsonb '[2]') -- will yield jsonb '{"a":[null,{"b":[1],"c":[2]}]}', but SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,b}', jsonb '[2]') -- will raise SQLSTATE 22023 (invalid_parameter_value): cannot replace existing key

刪除鍵(或索引)從JSON對象(或者,從數組)執行-操作員:

SELECT jsonb '{"a":1,"b":2}' - 'a', -- will yield jsonb '{"b":2}' jsonb '["a",1,"b",2]' - 1 -- will yield jsonb '["a","b",2]'

從JSON層次結構的深處刪除可以用#-

SELECT '{"a":[null,{"b":[3.14]}]}' #- '{a,1,b,0}' -- will yield jsonb '{"a":[null,{"b":[]}]}'

但可以使用以下方法直接聚合到json對象中,而不是聚合JSON字符串json_object_agg().

CREATE OR REPLACE FUNCTION "json_object_set_key"( "json" json, "key_to_set" TEXT, "value_to_set" anyelement ) RETURNS json LANGUAGE sql IMMUTABLE STRICT AS $function$ SELECT concat('{', string_agg(to_json("key") || ':' || "value", ','), '}')::json FROM (SELECT * FROM json_each("json") WHERE "key" <> "key_to_set" UNION ALL SELECT "key_to_set", to_json("value_to_set")) AS "fields" $function$;

SQLFiddle

一個版本,它設置多個鍵和值:

CREATE OR REPLACE FUNCTION "json_object_set_keys"( "json" json, "keys_to_set" TEXT[], "values_to_set" anyarray ) RETURNS json LANGUAGE sql IMMUTABLE STRICT AS $function$ SELECT concat('{', string_agg(to_json("key") || ':' || "value", ','), '}')::json FROM (SELECT * FROM json_each("json") WHERE "key" <> ALL ("keys_to_set") UNION ALL SELECT DISTINCT ON ("keys_to_set"["index"]) "keys_to_set"["index"], CASE WHEN "values_to_set"["index"] IS NULL THEN 'null'::json ELSE to_json("values_to_set"["index"]) END FROM generate_subscripts("keys_to_set", 1) AS "keys"("index") JOIN generate_subscripts("values_to_set", 1) AS "values"("index") USING ("index")) AS "fields" $function$;
CREATE OR REPLACE FUNCTION "json_object_update_key"( "json" json, "key_to_set" TEXT, "value_to_set" anyelement ) RETURNS json LANGUAGE sql IMMUTABLE STRICT AS $function$ SELECT CASE WHEN ("json" -> "key_to_set") IS NULL THEN "json" ELSE (SELECT concat('{', string_agg(to_json("key") || ':' || "value", ','), '}') FROM (SELECT * FROM json_each("json") WHERE "key" <> "key_to_set" UNION ALL SELECT "key_to_set", to_json("value_to_set")) AS "fields")::json END $function$;
CREATE OR REPLACE FUNCTION "json_object_set_path"( "json" json, "key_path" TEXT[], "value_to_set" anyelement ) RETURNS json LANGUAGE sql IMMUTABLE STRICT AS $function$ SELECT CASE COALESCE(array_length("key_path", 1), 0) WHEN 0 THEN to_json("value_to_set") WHEN 1 THEN "json_object_set_key"("json", "key_path"[l], "value_to_set") ELSE "json_object_set_key"( "json", "key_path"[l], "json_object_set_path"( COALESCE(NULLIF(("json" -> "key_path"[l])::text, 'null'), '{}')::json, "key_path"[l+1:u], "value_to_set" ) ) END FROM array_lower("key_path", 1) l, array_upper("key_path", 1) u $function$;
 
用戶回答回答於 2018-02-262018-02-26 17:03:04

 

UPDATE objects
SET body = jsonb_set(body, '{name}', '"Mary"', true) WHERE id = 1; 

其中主體是jsonb列類型。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM