如何修改新的PostgreSQL JSON数据类型内的字段?

使用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...

回答:

:在PostgreSQL

9.5中,jsonbPostgreSQL本身具有一些操纵功能(但对于json;没有;操纵json值需要强制转换)。

合并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>')

where <key>应该是字符串,并且<value>可以是任何to_jsonb()可接受的类型。

要 ,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]}]}'

为了 ,jsonb_insert()可以使用 该函数在9.6+中;仅在本节中,此函数 ):

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.ex。可以将JSON附加到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()pathin

target是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

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

jsonb '["a",1,"b",2]' - 1 -- will yield jsonb '["a","b",2]'

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

-- will yield jsonb '{"a":[null,{"b":[]}]}'

,您可以使用原始答案的修改版本(如下),但是可以使用直接将其聚合为JSON对象,而不是聚合JSON字符串json_object_agg()

:也可以在纯SQL中(没有plpython或plv8)(但需要9.3+,不能在9.2下使用)

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$;

:正如@ErwinBrandstetter所指出的,以上这些功能的工作方式类似于所谓的UPSERT(如果存在,则更新字段,如果不存在,则插入字段)。这是一个变体,仅UPDATE

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$;

:这是递归变体,可以设置(UPSERT)叶值(并使用此答案中的第一个函数),该叶值位于键路径(其中键只能引用内部对象,不支持内部数组):

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$;

更新:添加了用另一个给定密钥替换现有json字段的密钥的功能。在迁移或其他情况(如数据结构修改)中更新数据类型时可以派上用场。

CREATE OR REPLACE FUNCTION json_object_replace_key(

json_value json,

existing_key text,

desired_key text)

RETURNS json AS

$BODY$

SELECT COALESCE(

(

SELECT ('{' || string_agg(to_json(key) || ':' || value, ',') || '}')

FROM (

SELECT *

FROM json_each(json_value)

WHERE key <> existing_key

UNION ALL

SELECT desired_key, json_value -> existing_key

) AS "fields"

-- WHERE value IS NOT NULL (Actually not required as the string_agg with value's being null will "discard" that entry)

),

'{}'

)::json

$BODY$

LANGUAGE sql IMMUTABLE STRICT

COST 100;

:现在压缩函数。

以上是 如何修改新的PostgreSQL JSON数据类型内的字段? 的全部内容, 来源链接: utcz.com/qa/417354.html

回到顶部