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

使用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...
433552 次浏览

遗憾的是,我没有在文档中找到任何东西,但您可以使用一些变通方法,例如您可以编写一些扩展函数。

例如,在Python中:

CREATE or REPLACE FUNCTION json_update(data json, key text, value json)
returns json
as $$
from json import loads, dumps
if key is None: return data
js = loads(data)
js[key] = value
return dumps(js)
$$ language plpython3u

然后

update test set data=json_update(data, 'a', to_json(5)) where data->>'b' = '2';

更新: 使用PostgreSQL 9.5,在PostgreSQL本身中有一些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]}]}'

jsonb_insert()的完整参数列表:

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

同样,出现在path中的负整数从JSON数组的末尾开始计数。

所以,f.ex。附加到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$;

< a href = " http://sqlfiddle.com/ !15 / d41d8/2902 noreferrer“rel = > 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$;

指出上述函数的工作原理类似于所谓的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;

更新:函数现在被压缩。

在PostgreSQL 9.4中,我们实现了以下python函数。它也可以用于PostgreSQL 9.3。

create language plpython2u;


create or replace function json_set(jdata jsonb, jpaths jsonb, jvalue jsonb) returns jsonb as $$
import json


a = json.loads(jdata)
b = json.loads(jpaths)


if a.__class__.__name__ != 'dict' and a.__class__.__name__ != 'list':
raise plpy.Error("The json data must be an object or a string.")


if b.__class__.__name__ != 'list':
raise plpy.Error("The json path must be an array of paths to traverse.")


c = a
for i in range(0, len(b)):
p = b[i]
plpy.notice('p == ' + str(p))


if i == len(b) - 1:
c[p] = json.loads(jvalue)


else:
if p.__class__.__name__ == 'unicode':
plpy.notice("Traversing '" + p + "'")
if c.__class__.__name__ != 'dict':
raise plpy.Error("  The value here is not a dictionary.")
else:
c = c[p]


if p.__class__.__name__ == 'int':
plpy.notice("Traversing " + str(p))
if c.__class__.__name__ != 'list':
raise plpy.Error("  The value here is not a list.")
else:
c = c[p]


if c is None:
break


return json.dumps(a)
$$ language plpython2u ;

使用示例:

create table jsonb_table (jsonb_column jsonb);
insert into jsonb_table values
('{"cars":["Jaguar", {"type":"Unknown","partsList":[12, 34, 56]}, "Atom"]}');


select jsonb_column->'cars'->1->'partsList'->2, jsonb_column from jsonb_table;


update jsonb_table
set jsonb_column = json_set(jsonb_column, '["cars",1,"partsList",2]', '99');


select jsonb_column->'cars'->1->'partsList'->2, jsonb_column from jsonb_table;

请注意,我为以前的雇主写了一组C函数,用于将JSON数据作为文本(而不是jsonjsonb类型)操作PostgreSQL 7,8和9。例如,用json_path('{"obj":[12, 34, {"num":-45.67}]}', '$.obj[2]['num']')提取数据,用json_path_set('{"obj":[12, 34, {"num":-45.67}]}', '$.obj[2]['num']', '99.87')设置数据等等。它花了大约3天的时间,所以如果您需要在遗留系统上运行它,并且有空闲时间,那么这可能是值得的。我想C版本要比python版本快得多。

在@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"}}

下面的plpython代码片段可能会派上用场。

CREATE EXTENSION IF NOT EXISTS plpythonu;
CREATE LANGUAGE plpythonu;


CREATE OR REPLACE FUNCTION json_update(data json, key text, value text)
RETURNS json
AS $$
import json
json_data = json.loads(data)
json_data[key] = value
return json.dumps(json_data, indent=4)
$$ LANGUAGE plpythonu;


-- Check how JSON looks before updating


SELECT json_update(content::json, 'CFRDiagnosis.mod_nbs', '1')
FROM sc_server_centre_document WHERE record_id = 35 AND template = 'CFRDiagnosis';


-- Once satisfied update JSON inplace


UPDATE sc_server_centre_document SET content = json_update(content::json, 'CFRDiagnosis.mod_nbs', '1')
WHERE record_id = 35 AND template = 'CFRDiagnosis';

即使下面的函数不能满足这个请求(函数json_object_agg在PostgreSQL 9.3中不可用),下面的函数对于任何为PostgreSQL 9.4寻找||运算符的人来说都是有用的,因为在即将到来的PostgreSQL 9.5中实现了:

CREATE OR REPLACE FUNCTION jsonb_merge(left JSONB, right JSONB)
RETURNS JSONB
AS $$
SELECT
CASE WHEN jsonb_typeof($1) = 'object' AND jsonb_typeof($2) = 'object' THEN
(SELECT json_object_agg(COALESCE(o.key, n.key), CASE WHEN n.key IS NOT NULL THEN n.value ELSE o.value END)::jsonb
FROM jsonb_each($1) o
FULL JOIN jsonb_each($2) n ON (n.key = o.key))
ELSE
(CASE WHEN jsonb_typeof($1) = 'array' THEN LEFT($1::text, -1) ELSE '['||$1::text END ||', '||
CASE WHEN jsonb_typeof($2) = 'array' THEN RIGHT($2::text, -1) ELSE $2::text||']' END)::jsonb
END
$$ LANGUAGE sql IMMUTABLE STRICT;
GRANT EXECUTE ON FUNCTION jsonb_merge(jsonb, jsonb) TO public;
CREATE OPERATOR || ( LEFTARG = jsonb, RIGHTARG = jsonb, PROCEDURE = jsonb_merge );

在9.5中使用jsonb_set-

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

其中body是jsonb列类型。

在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)

要更新里面的字段,不要使用concat操作符||。请改用jsonb_set。这并不简单:

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

使用concat操作符{c,c1},例如:

UPDATE testjsonb SET object = object || '{"c":{"c1":"seedoctor"}}';

将移除{c,c2}和{c,c3}。

要获得更多的权力,请在Postgresql json函数文档中寻求权力。你可能会对#-操作符、jsonb_set函数和jsonb_insert函数感兴趣。

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

这似乎在PostgreSQL 9.5上有效

你也可以在jsonb中原子地增加键,如下所示:

UPDATE users SET counters = counters || CONCAT('{"bar":', COALESCE(counters->>'bar','0')::int + 1, '}')::jsonb WHERE id = 1;


SELECT * FROM users;


id |    counters
----+------------
1 | {"bar": 1}

>假设起始值为0。

有关更详细的解释,请参阅我的回答:https://stackoverflow.com/a/39076637

我为自己写了一个小函数,在Postgres 9.4中递归地工作。下面是函数(我希望它对你有用):

CREATE OR REPLACE FUNCTION jsonb_update(val1 JSONB,val2 JSONB)
RETURNS JSONB AS $$
DECLARE
result JSONB;
v RECORD;
BEGIN
IF jsonb_typeof(val2) = 'null'
THEN
RETURN val1;
END IF;


result = val1;


FOR v IN SELECT key, value FROM jsonb_each(val2) LOOP


IF jsonb_typeof(val2->v.key) = 'object'
THEN
result = result || jsonb_build_object(v.key, jsonb_update(val1->v.key, val2->v.key));
ELSE
result = result || jsonb_build_object(v.key, v.value);
END IF;
END LOOP;


RETURN result;
END;
$$ LANGUAGE plpgsql;

下面是使用示例:

select jsonb_update('{"a":{"b":{"c":{"d":5,"dd":6},"cc":1}},"aaa":5}'::jsonb, '{"a":{"b":{"c":{"d":15}}},"aa":9}'::jsonb);
jsonb_update
---------------------------------------------------------------------
{"a": {"b": {"c": {"d": 15, "dd": 6}, "cc": 1}}, "aa": 9, "aaa": 5}
(1 row)

正如你所看到的,它深入分析并在需要的地方更新/添加值。

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

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

希望它能帮助其他人!

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

如果您的字段类型是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函数和操作符

你可以在这里读到我的笔记

我发现以前的答案更适合有经验的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])

在“目标"-我将放置jsonb column-name(这是正在修改的表列)

&; __abc0 &;-是&;json keys路径;指向(并包括)我们将要覆盖的键

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

在本例中,我们希望更新位于key1下的key13的值(key1 ->键13):

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

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

对于那些使用mybatis的人,这里有一个更新语句示例:

<update id="saveAnswer">
update quiz_execution set answer_data = jsonb_set(answer_data, concat('{', #{qid}, '}')::text[], #{value}::jsonb), updated_at = #{updatedAt}
where id = #{id}
</update>
< p >
参数:< / >强

  • qid,字段的键。
  • value,是一个有效的json字符串,对于字段值,
    例如,通过jackson将对象转换为json字符串,
例如,我的字符串是这样的: {“a1":{“a11":“x",“a22":“y",“a33":“z"}} < / p >

我使用临时表更新jsons,这对于相当小的数据量(<1.000.000)已经足够了。我找到了另一条路,但后来去度假了,就忘了……

所以。查询将是这样的:

with temp_table as (
select
a.id,
a->'a1'->>'a11' as 'a11',
a->'a1'->>'a22' as 'a22',
a->'a1'->>'a33' as 'a33',
u1.a11updated
from foo a
join table_with_updates u1 on u1.id = a.id)
update foo a
set a = ('{"a1": {"a11": "'|| t.a11updated ||'",
"a22":"'|| t.a22 ||'",
"a33":"'|| t.a33 ||'"}}')::jsonb
from temp_table t
where t.id = a.id;

它与string的关系比json更大,但它是有效的。基本上,它将所有数据拉到临时表中,在用备份的数据插入concat漏洞时创建一个字符串,并将其转换为jsonb。

Json_set可能更有效,但我仍然掌握了它的窍门。当我第一次尝试使用它时,我完全把弦弄乱了……

你可以尝试更新如下:

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

举个例子:

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

UPDATE table_name SET attrs = jsonb_set(cast(attrs as jsonb), '{key}', '"new_value"', true) WHERE id = 'some_id';

这对我有用,attrs是一个json类型的字段。首先转换为jsonb,然后更新。

UPDATE table_name SET attrs = jsonb_set(cast(attrs as jsonb), '{key}', '"new_value"', true) WHERE attrs->>key = 'old_value';

你觉得这个解决办法怎么样?

它将添加新值或更新现有值。

编辑:编辑后使其与null和空对象一起工作

Edit2:编辑使它与对象中的对象一起工作…

create or replace function updateJsonb(object1 json, object2 json)
returns jsonb
language plpgsql
as
$$
declare
result jsonb;
tempObj1 text;
tempObj2 text;


begin
tempObj1 = substr(object1::text, 2, length(object1::text) - 2); --remove the first { and last }
tempObj2 = substr(object2::text, 2, length(object2::text) - 2); --remove the first { and last }


IF object1::text != '{}' and object1::text != 'null' and object1::text != '[]' THEN
result = ('{' || tempObj1 || ',' || tempObj2 || '}')::jsonb;
ELSE
result = ('{' || tempObj2 || '}')::jsonb;
END IF;
return result;
end;
$$;

用法:

update table_name
set data = updatejsonb(data, '{"test": "ok"}'::json)

如果你想在你的JSON更新命令中使用其他列的值,你可以使用字符串连接:

UPDATE table
SET column1 = column1::jsonb - 'key' || ('{"key": ' || column2::text ||  '}')::jsonb
where ...;

如果你想添加新的字段,你可以尝试:

typeorm代码

let keyName:string = '{key2}'
let valueName:string = '"new_value"'


emailLog: () => "jsonb_set(cast(email_log as jsonb), '" + keyNAme + "','" + valueName + "'," + "true" + ")"

这个解决方案是jsonb_set的替代方案,即使JSONB的列中有NULL也能工作。jsonb_set只在对象存在时起作用。

在下面的示例中,settings是用户表上的JSONB列。

 UPDATE public."Users"
SET settings = coalesce("settings", '{}')::jsonb || '{ "KeyToSet" : "ValueToSet" }'
WHERE id=35877;
select * from pg_settings where name = 'deadlock_timeout';
begin;
create temp table a2(data jsonb);
insert into a2 values('{
"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"
}
}}'::jsonb);
commit;

嵌套的jsonb结构更新。可应用于删除。

update a2 set data =
data::jsonb #-  '{key0, key1, key13}'
|| '{"key13":"screen labeling"}'::jsonb
returning *;

从PostgreSQL 14开始,你可以使用jsonb加下标直接访问JSON字段的元素,并最终更新它们。

UPDATE test SET data['a'] = '5' WHERE data['b'] = '2';