程序问答   发布时间:2022-06-02  发布网站:大佬教程  code.js-code.com
大佬教程收集整理的这篇文章主要介绍了如何修改新的PostgreSQL JSON数据类型内的字段?大佬教程大佬觉得挺不错的,现在分享给大家,也给大家做个参考。

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

开发过程中遇到如何修改新的PostgreSQL JSON数据类型内的字段?的问题如何解决?下面主要结合日常开发的经验,给出你关于如何修改新的PostgreSQL JSON数据类型内的字段?的解决方法建议,希望对你解决如何修改新的PostgreSQL JSON数据类型内的字段?有所启发或帮助;

在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()pathintarget是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
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 JSON数据类型内的字段?全部内容,希望文章能够帮你解决如何修改新的PostgreSQL JSON数据类型内的字段?所遇到的程序开发问题。

如果觉得大佬教程网站内容还不错,欢迎将大佬教程推荐给程序员好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。