如何修改新 PostgreSQL JSON 数据类型中的字段?
Posted
技术标签:
【中文标题】如何修改新 PostgreSQL JSON 数据类型中的字段?【英文标题】:How do I modify fields inside the new PostgreSQL JSON datatype? 【发布时间】:2013-08-15 02:24:55 【问题描述】:使用 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...
【问题讨论】:
【参考方案1】:遗憾的是,我在文档中没有找到任何内容,但您可以使用一些解决方法,例如您可以编写一些扩展函数。
例如,在 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';
【讨论】:
很遗憾 Amazon RDS 不支持 plpython3u!value
在设置非数字值(如字符串 (js[key] = loads(value)
) 时也需要 loads
- 否则:select json_update('"a":"a"', 'a', to_json('b')); -> "a": "\"b\""
这个答案也可以修改为包括在 value 设置为 None 时删除一个 key:`if value is None: del data[key]【参考方案2】:
更新:With 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+ 中;仅此函数,在本节中 em>):
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 数组的末尾开始计数。
所以,前任。可以通过以下方式追加到 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_object_agg()
聚合成一个 json 对象,而不是聚合一个 JSON 字符串。 p>
原答案:在纯 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 noted,上述这些函数的工作方式类似于所谓的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;
更新:函数现在被压缩了。
【讨论】:
我尝试了您的 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"
这相当于 UPSERT
,而不是 UPDATE
。如果 json 字段中尚不存在该键,则添加该键。查看这个相关问题以了解实际的UPDATE
:***.com/questions/7711432/…(这是针对复合类型,但主体与 json 类似。)
@ErwinBrandstetter 确实如此,但在 json 中,UPSERT 通常比类似 UPDATE 的修改更通用(考虑 f.ex. sqlfiddle.com/#!15/d41d8/2897 )——我解释了原来的问题是你如何使用 UPDATE 语句修改它们(json 列)?——除了单个条件可以将其转换为 UPDATE。
现在非常有用且完整。
@maxhud 取决于客户端(或您使用的客户端库)。如果可以,请使用显式类型(PostgreSQL 可以猜测参数化查询中的类型,但这通常不适用于多态函数)。但至少,您可以使用显式转换,例如 $2::text
。【参考方案3】:
在 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;
请注意,对于以前的雇主,我为 PostgreSQL 7、8 和 9 编写了一组 C 函数,用于将 JSON 数据作为文本(不是 json
或 jsonb
类型)进行操作。例如,提取数据用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版本快很多。
【讨论】:
【参考方案4】:基于@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$;
Recursive Delete By Key:通过 key-path 从 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"
【讨论】:
【参考方案5】:下面的 plpython sn-p 可能会派上用场。
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';
【讨论】:
【参考方案6】:即使下面的内容不能满足这个请求(函数 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 );
【讨论】:
【参考方案7】:在 9.5 中使用 jsonb_set-
UPDATE objects
SET body = jsonb_set(body, 'name', '"Mary"', true)
WHERE id = 1;
body 是 jsonb 列类型。
【讨论】:
嗨,为什么我不能像这样使用upper
:update objects set body=jsonb_set(body, 'name', upper('"Mary"'), true) where id=1;
它无法识别,或者我怎样才能实现相同的行为?谢谢
如果我要设置的值是另一列的子字符串而不是“玛丽”,我该怎么做?
@Andrew,您将使用to_jsonb(substring(column_name, 1, 2))
将列值转换为 jsonb。所以,update objects set body=jsonb_set(body, 'name', to_jsonb(upper(name_column)), true);
.【参考方案8】:
使用 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';
执行以下操作:
-
更新 a 字段
删除 b 字段
添加 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"');
例如对 c,c1 使用 concat 运算符:
UPDATE testjsonb SET object = object || '"c":"c1":"seedoctor"';
将删除 c,c2 和 c,c3。
如需更多权力,请在postgresql json functions documentation 寻求权力。有人可能对#-
运算符、jsonb_set
函数以及jsonb_insert
函数感兴趣。
【讨论】:
如果我必须更新两个字段,那么语法是什么? 如果我有一个带有字段名称的 json 列,如何将 lastname 字段添加到该列中 应该清楚:UPDATE users SET profile = profile || '"lastname":"Washington"' WHERE profile->>'name' = 'George Washington';
【参考方案9】:
UPDATE test
SET data = data::jsonb - 'a' || '"a":5'::jsonb
WHERE data->>'b' = '2'
这似乎适用于 PostgreSQL 9.5
【讨论】:
对我有用,据我所知,这会从数据中删除字段“a”,然后在字段“a”中附加新值。在我的例子中,“a”的值是基于一列的。更新测试集数据 = data::jsonb - 'a' || ('"a":"'|| myColumn || '"')::jsonb;【参考方案10】:您还可以像这样在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://***.com/a/39076637
【讨论】:
【参考方案11】:我为自己编写了一个在 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)
如您所见,它会深入分析并在需要时更新/添加值。
【讨论】:
【参考方案12】:当我尝试更新字符串类型字段时,这对我有用。
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 特定函数进行操作一样【参考方案13】:如果您的字段类型是 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 Functions and Operators
You can read my note here
【讨论】:
更新 JSON 字段的简单和更好的方法,如果您不担心属性顺序重新排列。【参考方案14】:我发现以前的答案更适合有经验的 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 (key1 -> key13) 下的 key13 的值:
因此路径语法是:'key1,key13' (路径是最难破解的部分——因为教程很糟糕)
jsonb_set(jsonb_column,'key1,key13','"See main screen labeling"')
【讨论】:
【参考方案15】:对于那些使用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>
参数:
qid
,字段的键。
value
,是一个有效的 json 字符串,用于字段值,
例如通过jackson
从对象转换为json字符串,
【讨论】:
【参考方案16】:因此,例如,我的字符串如下所示: "a1":"a11":"x","a22":"y","a33":"z"
我使用临时表更新 jsons,这对于相当少量的数据(
所以。查询将是这样的:
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;
与字符串相比,它与 json 有更多的关系,但它确实有效。基本上,它将所有数据拉入临时表,创建一个字符串,同时用您备份的数据插入连接孔,并将其转换为 jsonb。
Json_set 可能更有效,但我仍然掌握它。我第一次尝试使用它时,我完全搞砸了字符串......
【讨论】:
嗨,欢迎来到 ***!请注意,这个问题已经有一个公认的答案。 请注意,此方法可能存在安全漏洞,具体取决于字段的内容,因为您需要将数据编码为 JSON 字符串,否则攻击者可以操纵您的数据结构。【参考方案17】:您可以尝试如下更新:
语法: UPDATE table_name SET column_name = column_name::jsonb || '"key":new_value' WHERE column_name 条件;
以您为例:
更新测试集数据 = data::jsonb || '"a":new_value' WHERE 数据->>'b' = '2';
【讨论】:
【参考方案18】: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';
【讨论】:
【参考方案19】:你觉得这个解决方案怎么样?
它将添加新值或更新现有值。
编辑:编辑使其适用于 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)
【讨论】:
我认为这个实现很可能存在安全漏洞。如果攻击者可以影响tempObj1
或tempObj2
的内容以包含引号,他或她可以通过在字符串中具有合适的值来修改对象的结构,因为代码将生成的字符串重新解释为jsonb。【参考方案20】:
如果您想在 JSON 更新命令中使用其他列的值,您可以使用字符串连接:
UPDATE table
SET column1 = column1::jsonb - 'key' || ('"key": ' || column2::text || '')::jsonb
where ...;
【讨论】:
【参考方案21】:如果您也想添加新字段,您可以尝试:
typeorm 代码
let keyName:string = 'key2'
let valueName:string = '"new_value"'
emailLog: () => "jsonb_set(cast(email_log as jsonb), '" + keyNAme + "','" + valueName + "'," + "true" + ")"
【讨论】:
以上是关于如何修改新 PostgreSQL JSON 数据类型中的字段?的主要内容,如果未能解决你的问题,请参考以下文章
如何使用新的 PostgreSQL JSON 数据类型中的字段进行查询?
如何在 Django 中使用 PostgreSQL 9.2 JSON 数据类型?
C# 用Newtonsoft.Json 修改 添加 JSON
如何在 Postgresql 中添加、更新和删除 Json 数据类型?