Як змінити поля всередині нового типу даних 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...

Відповіді:


331

Оновлення : з PostgreSQL 9.5 є деякі jsonbфункції маніпуляції в самому PostgreSQL (але жоден для 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

Однак ця функція працює дещо інакше (ніж jsonb_set()), коли pathв 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().

Оригінальна відповідь : Можливо (без plpython або plv8) і в чистому SQL (але потрібно 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

1
Це виконує еквівалент an UPSERT, а не an UPDATE. Якщо ключ ще не існує в полі json, він додається. Подивіться на це відповідне запитання для фактичного UPDATE: stackoverflow.com/questions/7711432/… (Це для складеного типу, але головна схожа на json.)
Ервін Брандстеттер

1
@ErwinBrandstetter це правда, але в json UPSERT, як правило, більш загальна, ніж модифікація, подібна до UPDATE (розгляньте f.ex. sqlfiddle.com/#!15/d41d8/2897 ) - я інтерпретував оригінальне запитання як те, як ви змінюєте їх (стовпці json) за допомогою оператора UPDATE? - крім того, одна умова може перетворити це на ОНОВЛЕННЯ.
pozs

1
Зараз дуже корисно і повно.
Erwin Brandstetter

1
@maxhud, що залежить від клієнта (або клієнтської бібліотеки, яку ви використовуєте). Якщо можете, використовуйте явні типи (PostgreSQL може вгадати типи в параметризованих запитах, але це, як правило, не добре працює з поліморфними функціями). Але принаймні, ви можете використовувати явні касти, наприклад $2::text.
pozs

98

З 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;він не розпізнає, або як я можу досягти такої ж поведінки? thx
Рафаель Капучо

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функція.


і якщо мені доведеться оновити два поля, то що таке синтаксис?
Суніл Гарг

якщо у мене стовпчик json з назвою поля, як додати поле прізвища до цього стовпця
Bionix1441

Повинно бути зрозумілим:UPDATE users SET profile = profile || '{"lastname":"Washington"}' WHERE profile->>'name' = 'George Washington';
Фанді Сусанто

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 за ключем. (потрібна json_object_set_keyфункція @ pozs )

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

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

Здається, це працює над PostgreSQL 9.5


Для мене працює, наскільки я зрозумів, це видалити поле "a" з даних, а потім додати поле "a" з новим значенням. У моєму випадку значення "а" базувалося на стовпчику. ОНОВЛЕННЯ тестових даних SET = data :: jsonb - 'a' || ('{"a": "' || myColumn || '"}') :: jsonb;
sebge2

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, якщо ви не турбуєтесь про перестановки порядку власності.
Картік Сіварадж

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 у вигляді тексту (а не як jsonабо jsonbтипу) для 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 днів, тож якщо вам потрібно, щоб вона працювала на застарілих системах і у вас був час на запас, можливо, варто докласти зусиль. Я думаю, версія С набагато швидша, ніж версія python.


2

Незважаючи на те, що наступне не задовольнить цей запит (функція 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 );

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 є стовпець jsonb з іменем body, і ви хочете змінити body.some_key = 'value'


на жаль, це переформатує JSON так само, як маніпуляції через специфічні функції JSON
Lu55,

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

2
Також valueбуде потрібно a loadsпри встановленні нечислових значень, таких як рядки ( js[key] = loads(value)) - інакше:select json_update('{"a":"a"}', 'a', to_json('b')); -> {"a": "\"b\""}
hooblei

Цю відповідь можна також змінити, щоб вона включала видалення ключа, коли для параметра встановлено значення None: `якщо значення є None: del data [ключ]
Джошуа Бернс

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 (), щоб призначити нове значення ключу13

параметри до jsonb_set ()

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

у " target " - я поміщу ім'я стовпця jsonb (це стовпець таблиці, що змінюється)

" шлях " - це "шлях ключів json", що веде до (і включає) ключ, який ми збираємося перезаписати

" new_value " - це нове значення, яке ми присвоюємо

у нашому випадку ми хочемо оновити значення key13, яке знаходиться під key1 (key1 -> 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 для значення поля,
    наприклад, перетвореного з об'єкта в рядок json через jackson,

0

Так, наприклад, мій рядок виглядає так: {"a1": {"a11": "x", "a22": "y", "a33": "z"}}

Я оновлюю jsons за допомогою таблиці temp, що досить добре для досить невеликої кількості даних (<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, але він працює. В основному, він перетягує всі дані в темп-таблицю, створює рядок під час підключення концентрованих отворів із резервними копіями даних та перетворює їх у jsonb.

Json_set може бути більш ефективним, але я все-таки переглядаю це. Перший раз, коли я спробував це використати, я повністю переплутав струну ...


1
привіт і ласкаво просимо до StackOverflow! Зауважте, що відповідь на це питання вже є.
hongsy

-2

Якщо ви робите цей запит із клієнтом мови програмування, наприклад, з python pycopg2або Node Postgres, переконайтеся, що спочатку проаналізуйте нові дані в JSON.

Це може легко виглядати як словник python - це те саме, що об'єкт JSON, але він спочатку не робить json.dumps у словнику.

Простий фрагмент пітона:

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

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.