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


235

使用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...

Answers:


330

更新在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()可接受的类型。

在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附加到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()pathin target是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

可以使用-运算符从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
  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$;

更新:现在压缩函数。


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 2014年

1
这相当于UPSERT,而不是UPDATE。如果密钥在json字段中尚不存在,则将其添加。实际看一下这个相关问题UPDATEstackoverflow.com/questions/7711432/…(这是一种复合类型,但是json的原理相似。)
Erwin Brandstetter 2014年

1
@ErwinBrandstetter是正确的,但在JSON中,UPSERT 通常比类似UPDATE的修改更通用(请考虑f.ex. sqlfiddle.com/#!15/d41d8/2897)-我将原始问题解释为如何修改他们(使用UPDATE语句)(json列)?-除了单个条件,还可以将其转换为UPDATE。
pozs

1
现在非常有用且完整。
Erwin Brandstetter 2014年

1
取决于客户端(或您使用的客户端库)的@maxhud。如果可以,请使用显式类型(PostgreSQL可以在参数化查询中猜测类型,但通常不适用于多态函数)。但至少,您可以使用显式强制转换,例如$2::text
pozs 2015年

98

在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;它无法识别,或者我该如何实现相同的行为?thx
Rafael Capucho

1
如果我要设置的值是另一列的子字符串而不是“ Mary”,我该怎么办?
安德鲁

58

使用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. 更新一个字段
  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功能感兴趣。


如果我必须更新两个字段,那么语法是什么?
Sunil Garg '18

如果我有一个带有字段名称的json列,如何将姓氏字段添加到此列
Bionix1441

应该清楚:UPDATE users SET profile = profile || '{"lastname":"Washington"}' WHERE profile->>'name' = 'George Washington';
Fandi Susanto

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

很有用!谢谢。
1111161171159459134 2015年

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

这似乎可以在PostgreSQL 9.5上工作


就我所知,对我有用,这将从数据中删除字段“ a”,然后将字段“ a”附加新值。在我的情况下,“ a”的值基于一列。UPDATE测试SET数据= data :: jsonb-'a'|| ('{“ a”:“'|| myColumn ||'”}'):: jsonb;
sebge2 '18

7

如果您的字段类型为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

4

在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数据作为文本(而不是a jsonjsonbtype)用于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版本快得多。


2

即使以下内容不能满足该请求(PostgreSQL 9.3中不提供函数json_object_agg),以下内容对于正在寻找||的任何人也可能有用。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 );

2

我为自己编写了一个小的函数,该函数在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)

如您所见,它会深入分析并在需要时更新/添加值。


2

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

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进行了重新格式化
Lu55,18年

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!
dbau 2014年

2
value也将需要loads设置非数值时喜欢串(js[key] = loads(value):) -否则select json_update('{"a":"a"}', 'a', to_json('b')); -> {"a": "\"b\""}
hooblei

还可以修改此答案,包括在将值设置为None时删除密钥:`如果value为None:del data [key]
Joshua Burns

1

以下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';

1

我发现以前的答案适合有经验的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列名(这是正在修改的表列)

path ”-是导致(并包括)我们将要覆盖的密钥的“ json密钥路径”

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

在我们的例子中,我们要更新位于key1(key1-> key13)下的key13的值:

因此,路径语法为:'{key1,key13}'(路径是最难破解的部分-因为教程太糟糕了)

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

0

您还可以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


0

对于使用的用户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,从object转换为json字符串

0

因此,例如,我的字符串如下所示:{“ a1”:{“ a11”:“ x”,“ a22”:“ y”,“ a33”:“ z”}}

我通过使用临时表来更新json,对于足够少的数据量(<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;

它与字符串比json有更多关系,但它可以工作。基本上,它将所有数据提取到临时表中,在使用您备份的数据插入concat漏洞的同时创建一个字符串,然后将其转换为jsonb。

Json_set可能会更有效,但是我仍然可以理解。第一次尝试使用它时,我完全弄乱了琴弦...


1
嗨,欢迎来到StackOverflow!请注意,此问题已被接受。
hongsy

-2

如果要使用编程语言客户端(例如来自python pycopg2或)进行此查询Node Postgres,请确保首先将新数据解析为JSON。

它很容易看起来像python字典,它与JSON对象相同,但首先不对字典进行json.dumps。

一个简单的python代码段:

def change_destination(self,parcel_id,destlatlng): query="UPDATE parcels SET destlatlng = '{}' WHERE parcel_id ={};".format(json.dumps(destlatlng), parcel_id) self.cursor.execute(query2) self.connection.commit()

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.