栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 面试经验 > 面试问答

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

面试问答 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

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

更新 :在PostgreSQL
9.5中

jsonb
PostgreSQL本身具有一些操纵功能(但对于
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()
可接受的类型。

在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,          pathtext[],          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()
可以使用 该函数在9.6+中;仅在本节中,此函数 ):

SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,b,0}', jsonb '2')-- will yield jsonb '{"a":[null,{"b":[2,1]}]}', andSELECT 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()
path
in
target
是JSON对象的键时,此函数的工作原理(与)略有不同。在这种情况下,只会在不使用键时为JSON对象添加一个新的键值对。如果使用它,将引发错误:

SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,c}', jsonb '[2]')-- will yield jsonb '{"a":[null,{"b":[1],"c":[2]}]}', butSELECT 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":[]}]}'

对于9.4
,您可以使用原始答案的修改版本(如下),但是可以使用直接将其聚合为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  STRICTAS $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  STRICTAS $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$;

编辑2
:正如@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  STRICTAS $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")::jsonEND$function$;

编辑3 :这是递归变体,可以设置(

UPSERT
)叶值(并使用此答案中的第一个函数),该叶值位于键路径(其中键只能引用内部对象,不支持内部数组):

CREATE OR REPLACE FUNCTION "json_object_set_path"(  "json"          json,  "key_path"      TEXT[],  "value_to_set"  anyelement)  RETURNS json  LANGUAGE sql  IMMUTABLE  STRICTAS $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;

更新 :现在压缩函数。



转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/369205.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号