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

380

使用postgresql 9.3,我可以使用SELECT选择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...
24个回答

499
更新: PostgreSQL 9.5中,内置了一些jsonb操作功能(但是没有针对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>')

<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()函数(在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]}]}'
< p > jsonb_insert()的完整参数列表:

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

再次提醒,出现在path中的负整数是从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

然而,当target中的path是JSON对象的键时,此函数与jsonb_set()有些不同。在这种情况下,仅当该键未被使用时,它才会为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":[]}]}'

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

原始答案:在纯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$;

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

编辑3: 这里是递归版本,可以在键路径上设置(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;

更新:现在函数已经被压缩。


5
我尝试了你的plpgsql函数,但不确定如何使用它 - 当我尝试 select json_object_set_key((select data from test where data->>'b' = '2'), 'b', 'two'); 时,出现了错误。 错误信息是“ERROR: could not determine polymorphic type because input has type "unknown"(无法确定多态类型,因为输入具有未知类型)”。 - user9645
1
这执行等同于**UPSERT**的操作,而不是UPDATE。如果json字段中的键尚不存在,则将其添加。请参考以下相关问题以获取实际的UPDATE:https://dev59.com/Imsz5IYBdhLWcg3wsaPN(这是针对复合类型,但原则类似于json)。 - Erwin Brandstetter
2
@ErwinBrandstetter 这是真的,但在json中,UPSERT 通常 比UPDATE类似的修改更普遍(考虑例如http://sqlfiddle.com/#!15/d41d8/2897)--我将原始问题解释为*如何使用UPDATE语句修改它们(json列)?*--此外,一个单一的条件可以将其转换为UPDATE。 - pozs
1
非常有用且完整。 - Erwin Brandstetter
1
@maxhud 这取决于客户端(或您使用的客户端库)。如果可以的话,请使用显式类型(PostgreSQL可以猜测参数化查询中的类型,但这通常与多态函数不兼容)。但至少,您可以使用显式转换,例如 $2::text - pozs
显示剩余20条评论

183

使用9.5版本,可以使用jsonb_set函数-

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

其中 body 是一个 jsonb 列类型。


嗨,为什么我不能像这样使用 upperupdate objects set body=jsonb_set(body, '{name}', upper('"Mary"'), true) where id=1; 它无法识别,或者我该如何实现相同的行为?谢谢。 - Rafael Capucho
1
如果我想设置的值是另一列的子字符串而不是“Mary”,我该怎么做? - Andrew
2
@Andrew,你可以使用to_jsonb(substring(column_name, 1, 2))将列的值转换为jsonb。因此,update objects set body=jsonb_set(body, '{name}', to_jsonb(upper(name_column)), true); - Mr. Curious

106
2023年8月更新:
除了下面的信息外,还请参考Kouber Saparev关于PostgreSQL 14的回答。
使用Postgresql 9.5可以按照以下步骤完成:
UPDATE test
SET data = data - 'a' || '{"a":5}'
WHERE data->>'b' = '2';

或者

UPDATE test
SET data = jsonb_set(data, '{a}', '5'::jsonb);

有人问如何一次性更新jsonb值中的多个字段。假设我们创建了一个表:

CREATE TABLE testjsonb ( id SERIAL PRIMARY KEY, object JSONB );

然后我们插入一行实验数据:
INSERT INTO testjsonb
VALUES (DEFAULT, '{"a":"one", "b":"two", "c":{"c1":"see1","c2":"see2","c3":"see3"}}');

然后我们更新该行:
UPDATE testjsonb SET object = object - 'b' || '{"a":1,"d":4}';

以下是:
  1. 更新字段a
  2. 删除字段b
  3. 添加字段d
选择数据:
SELECT jsonb_pretty(object) FROM testjsonb;

将导致:

      jsonb_pretty
-------------------------
 {                      +
     "a": 1,            +
     "c": {             +
         "c1": "see1",  +
         "c2": "see2",  +
         "c3": "see3",  +
     },                 +
     "d": 4             +
 }
(1 row)

要更新内部字段,请不要使用连接运算符||。而是使用jsonb_set,这并不简单:

UPDATE testjsonb SET object =
jsonb_set(jsonb_set(object, '{c,c1}','"seeme"'),'{c,c2}','"seehim"');

使用连接运算符 {c,c1} 例如:
UPDATE testjsonb SET object = object || '{"c":{"c1":"seedoctor"}}';

将删除 {c,c2} 和 {c,c3}。
要获得更多的功能,请查看postgresql json functions documentation。可能会对 #- 运算符、jsonb_set 函数和 jsonb_insert 函数感兴趣。

如果我需要更新两个字段,那么语法是什么? - Sunil Garg
如果我有一个带有字段名称的JSON列,我该如何向该列添加lastname字段? - Bionix1441
1
应该很清楚:UPDATE users SET profile = profile || '{"lastname":"Washington"}' WHERE profile->>'name' = 'George Washington'; - Fandi Susanto

20

如果您的字段类型为json,则以下内容适用于您。

UPDATE 
table_name
SET field_name = field_name::jsonb - 'key' || '{"key":new_val}' 
WHERE field_name->>'key' = 'old_value'.

运算符“-”从左操作数中删除键/值对或字符串元素。 键/值对基于其键值进行匹配。

运算符'||'将两个jsonb值连接成一个新的jsonb值。

由于这些是jsonb运算符,您只需要强制转换为::jsonb

更多信息:JSON函数和运算符

您可以在此处阅读我的笔记


如果您不担心属性顺序的重新排列,更新JSON字段的简单和更好的方法。 - Karthik Sivaraj

18
自PostgreSQL 14版本起,您可以使用jsonb下标操作符直接访问JSON字段的元素,并对其进行更新。 了解更多
UPDATE test SET data['a'] = '5' WHERE data['b'] = '2';

14
UPDATE test
SET data = data::jsonb - 'a' || '{"a":5}'::jsonb
WHERE data->>'b' = '2'

这似乎在 PostgreSQL 9.5 上运行良好。


1
就我所理解的,这对我来说是有效的,它会从数据中删除字段"a",然后将字段"a"与新值附加。在我的情况下,“a”的值基于一列。UPDATE test SET data = data::jsonb - 'a' || ('{"a":"'|| myColumn || '"}')::jsonb; - sebge2

12

我发现之前的回答更适合有经验的PostgreSQL用户。这个回答是给初学者的:

假设您有一个类型为JSONB的表列,其值如下:

{
    "key0": {
        "key01": "2018-05-06T12:36:11.916761+00:00",
        "key02": "DEFAULT_WEB_CONFIGURATION",
        
    "key1": {
        "key11": "Data System",
        "key12": "<p>Health,<p>my address<p>USA",
        "key13": "*Please refer to main screen labeling"
    }
}

假设我们想在行中设置一个新值:

"key13": "*Please refer to main screen labeling"

而是放置该值:

"key13": "See main screen labeling"

我们使用json_set()函数为key13分配一个新值。

jsonb_set()函数的参数。

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

在 "target" 中 - 我将放置jsonb列名(这是正在修改的表列)

"path"- 是通往(包括)我们要覆盖的键的“json键路径”

"new_value" - 这是我们分配的新值

在我们的情况下,我们想要更新位于key1下的key13的值(key1-> key13):

因此路径语法是:'{key1,key13}'(路径是最棘手的部分 - 因为教程很糟糕)

jsonb_set(jsonb_column,'{key1,key13}','"See main screen labeling"')

11

当我尝试更新一个字符串类型的字段时,这对我很有用。

UPDATE table_name 
SET body = jsonb_set(body, '{some_key}', to_json('value'::text)::jsonb);

希望对其他人有所帮助!

假设表table_name有一个名为body的jsonb列,并且您想要更改body.some_key = 'value'


不幸的是,这会以与通过JSON特定函数进行操作相同的方式重新格式化JSON。 - Ilya Serbis

11

您可以尝试以下更新:

语法: UPDATE table_name SET column_name = column_name::jsonb || '{"key":new_value}' WHERE column_name condition;

以您的示例为例:

UPDATE test SET data = data::jsonb || '{"a":new_value}' WHERE data->>'b' = '2';


9

在 @pozs 的答案基础上,这里提供了一些可能对某些人有用的 PostgreSQL 函数。(需要 PostgreSQL 9.3+)

按键删除:通过键名从 JSON 结构中删除一个值。

CREATE OR REPLACE FUNCTION "json_object_del_key"(
  "json"          json,
  "key_to_del"    TEXT
)
  RETURNS json
  LANGUAGE sql
  IMMUTABLE
  STRICT
AS $function$
SELECT CASE
  WHEN ("json" -> "key_to_del") IS NULL THEN "json"
  ELSE (SELECT concat('{', string_agg(to_json("key") || ':' || "value", ','), '}')
          FROM (SELECT *
                  FROM json_each("json")
                 WHERE "key" <> "key_to_del"
               ) AS "fields")::json
END
$function$;

按键递归删除:通过键路径从JSON结构中删除一个值。(需要@pozs的json_object_set_key函数)

CREATE OR REPLACE FUNCTION "json_object_del_path"(
  "json"          json,
  "key_path"      TEXT[]
)
  RETURNS json
  LANGUAGE sql
  IMMUTABLE
  STRICT
AS $function$
SELECT CASE
  WHEN ("json" -> "key_path"[l] ) IS NULL THEN "json"
  ELSE
     CASE COALESCE(array_length("key_path", 1), 0)
         WHEN 0 THEN "json"
         WHEN 1 THEN "json_object_del_key"("json", "key_path"[l])
         ELSE "json_object_set_key"(
           "json",
           "key_path"[l],
           "json_object_del_path"(
             COALESCE(NULLIF(("json" -> "key_path"[l])::text, 'null'), '{}')::json,
             "key_path"[l+1:u]
           )
         )
       END
    END
  FROM array_lower("key_path", 1) l,
       array_upper("key_path", 1) u
$function$;

使用示例:

s1=# SELECT json_object_del_key ('{"hello":[7,3,1],"foo":{"mofu":"fuwa", "moe":"kyun"}}',
                                 'foo'),
            json_object_del_path('{"hello":[7,3,1],"foo":{"mofu":"fuwa", "moe":"kyun"}}',
                                 '{"foo","moe"}');

 json_object_del_key |          json_object_del_path
---------------------+-----------------------------------------
 {"hello":[7,3,1]}   | {"hello":[7,3,1],"foo":{"mofu":"fuwa"}}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接