Как изменить поля в новом типе данных PostgreSQL JSON?

С postgresql 9.3 я могу ВЫБРАТЬ определенные поля типа данных JSON, но как их изменить с помощью ОБНОВЛЕНИЯ? Я не могу найти никаких примеров этого в документации 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...

person user9645    schedule 13.08.2013    source источник


Ответы (20)


Обновление: с 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.

Итак, например, добавление в конец массива 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$;

Обновлено: добавлена ​​функция для замены существующего ключа 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;

Обновление: функции сжаты.

person pozs    schedule 06.05.2014
comment
Я попробовал вашу функцию 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" - person user9645; 07.05.2014
comment
@ user9645 Следует использовать как select json_object_set_key('{"b":2}', 'b', 'two'::text). Ваша проблема в том, что 'two' имеет тип unknown, если вы вызываете его с помощью литерала. То же самое с вызовом select to_json('two'). Эта функция принимает в качестве значения любой тип и преобразуется в json с to_json(), но это имеет такую ​​стоимость, что этот тип не может быть неизвестен. - person pozs; 07.05.2014
comment
Кстати, это не plpgsql. Это чистый SQL. - person pozs; 07.05.2014
comment
Спасибо @pozs - это работает для меня и не требует Python / Perl / и т. Д. это бонус. - person user9645; 20.05.2014
comment
Это эквивалентно UPSERT, а не UPDATE. Если ключа еще нет в поле json, он добавляется. Посмотрите в этом связанном вопросе фактическое название UPDATE: stackoverflow.com/questions/7711432/ (Это для составного типа, но принцип аналогичен для json.) - person Erwin Brandstetter; 13.08.2014
comment
@ErwinBrandstetter - это правда, но в json UPSERT обычно более общий, чем модификация, подобная UPDATE (рассмотрим f.ex. sqlfiddle.com / #! 15 / d41d8 / 2897) - я интерпретировал исходный вопрос как как изменить их (столбцы json) с помощью ОБНОВЛЕНИЯ оператор? - кроме того, одно условие может преобразовать это в ОБНОВЛЕНИЕ. - person pozs; 13.08.2014
comment
Очень полезно и полно. - person Erwin Brandstetter; 14.08.2014
comment
Это довольно полезный набор функций, для написания которого, очевидно, потребовалось много навыков и размышлений. Большое спасибо, @pozs! - person Dan Lenski; 23.09.2014
comment
Хорошие функции, можно ли их использовать для сложных объектов? Например. select json_object_set_key('{"A":{"a":1},"B":{"b":2}}'::json, '{"B"->"b"}', 'two'::text); не изменяет клавишу B- ›b ... - person user9645; 29.10.2014
comment
@ user9645 ключ здесь - это просто ключ строки (текста), а не сложный ключ path. Однако эту функцию можно использовать для того, что вы хотите, но вы должны применить ее дважды, например: select json_object_set_key(j, 'B', json_object_set_key(j->'B', 'b', 'two'::text)) from values('{"A":{"a":1},"B":{"b":2}}'::json) v(j) (я знаю, что это далеко не оптимально, но с этим шаблоном можно создать рекурсивную функцию) - person pozs; 30.10.2014
comment
@pozs Спасибо, но мне нужно общее решение, думаю, я укусит пулю и воспользуюсь расширением python. - person user9645; 06.11.2014
comment
Надеюсь, это будет встроенная функция для postgresql 9.4 '___') - person Kokizzu; 28.11.2014
comment
Как вы используете это с обновлением sql? 'UPDATE entry SET data = json_object_set_key (data, $ 1, $ 2)' не работает - выдает эту ошибку: не удалось определить полиморфный тип - person maxhud; 25.04.2015
comment
@maxhud, который зависит от клиента (или клиентской библиотеки, которую вы используете). По возможности используйте явные типы (PostgreSQL может угадывать типы в параметризованных запросах, но обычно это плохо работает с полиморфными функциями). Но, по крайней мере, вы можете использовать явное приведение типов, например $2::text. - person pozs; 27.04.2015
comment
спасибо - @pozs было бы неплохо добавить это в ответ! Если кому-то еще это нужно, это работает с PHP: pg_query_params("UPDATE entry SET data = json_object_set_key(data, $1::text, $2::json)", array($key, $value)) - person maxhud; 27.04.2015
comment
@pozs, пожалуйста, можно один для удаления ключевого пути? - person Reza S; 14.05.2015
comment
@RezaS, вы можете создать его, используя рекурсивный вариант (json_object_set_path), просто используйте функцию удаления из моего другого ответа (stackoverflow.com/ a / 23491408/1499698) в случае WHEN 1 (и вы можете использовать что-то вроде 'null'::json в случае WHEN 0, если вы хотите предоставить что-то значимое, когда вызывается с пустым ключевым путем (и, очевидно, вы не не нужен параметр value_to_set). - person pozs; 14.05.2015
comment
@pozs Спасибо, чувак, я все-таки сделал это, думал немного иначе. Я создал json_object_del_key, и json_object_del_path вскоре опубликует SQL-запросы к этому ответу. - person Reza S; 14.05.2015
comment
здесь также следует добавить обновление определенных значений в массиве. - person Thellimist; 30.08.2015
comment
@pozs Большое спасибо! ты спас мне день. Я застрял на Postgres v9.4, поэтому я попробовал ваши сценарии, и он работает отлично! - person vichsu; 02.02.2017
comment
@pozs привет, вы понимаете, почему следующее ничего не возвращает, когда есть 100% ключ с именем currency? Customer.where(id: 52).select("metadata::jsonb - 'currency'"). Если я просто выберу Customer.where(id: 52).select("metadata::jsonb"), он вернет весь хеш: [#<Customer:0x007fe02144f800 id: nil, metadata: {"currency"=>"EUR", "api_enabled"=>false, "scenario_ids"=>[3, 4], "system_patching_enabled"=>false}>] - person Andrey Deineko; 09.03.2017
comment
@AndreyDeineko Я не так хорошо знаком с рубином, но если результатом будет экземпляр клиента, тогда, возможно, вам также следует использовать псевдоним столбца, например Customer.where(id: 52).select("metadata::jsonb - 'currency' AS metadata") - без псевдонима столбец будет иметь произвольное имя, если его значение рассчитано. - person pozs; 09.03.2017
comment
привет, а что, если в данных json есть массив? бывший. {категория: {цвет: [{синий}, {желтый}]}} @ErwinBrandstetter - person john igneel; 30.05.2017
comment
Пути @johnigneel в jsonb_set() также могут содержать индексы массива JSON. Кроме того, для 9.6+ есть специальная функция jsonb_insert(), если вы хотите расширить массив (хороший вопрос, я скоро обновлю ответ). - Для более ранних версий было бы довольно сложно ввести поддержку индекса массива (но теоретически это возможно). - person pozs; 30.05.2017
comment
@pozs jsonb_set () звучит хорошо, но грустно. Я использую более раннюю версию postgres (9.3). Мне интересно, как это можно реализовать внутри функции для обновления в массиве в json. - person john igneel; 01.06.2017

С 9.5 используйте jsonb_set-

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

где body - тип столбца jsonb.

person Teo Choong Ping    schedule 21.01.2016
comment
Привет, почему я не могу использовать upper так: update objects set body=jsonb_set(body, '{name}', upper('"Mary"'), true) where id=1; он не распознает, или как я могу добиться такого же поведения? Спасибо - person Rafael Capucho; 19.12.2016
comment
Если значение, которое я хочу установить, является подстрокой из другого столбца, а не из Мэри, как мне это сделать? - person Andrew; 16.05.2017
comment
@Andrew, вы должны использовать to_jsonb(substring(column_name, 1, 2)) для преобразования значения столбца в jsonb. Итак, update objects set body=jsonb_set(body, '{name}', to_jsonb(upper(name_column)), true);. - person Mr. Curious; 13.04.2021

В Postgresql 9.5 это можно сделать, выполнив следующие действия:

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

OR

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.

person Fandi Susanto    schedule 27.06.2016
comment
и если мне нужно обновить два поля, то каков синтаксис? - person Sunil Garg; 28.04.2018
comment
если у меня есть столбец json с именем поля, как мне добавить поле фамилии в этот столбец - person Bionix1441; 08.11.2018
comment
Должно быть ясно: UPDATE users SET profile = profile || '{"lastname":"Washington"}' WHERE profile->>'name' = 'George Washington'; - person Fandi Susanto; 12.11.2018

Если ваш тип поля 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

Вы можете прочитать мою заметку здесь

person Neethu    schedule 03.05.2018
comment
Простой и лучший способ обновить поля JSON, если вас не беспокоит изменение порядка свойств. - person Karthik Sivaraj; 06.02.2020

Чтобы опираться на ответы @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"}}
person shru    schedule 14.05.2015
comment
Очень полезный! Спасибо. - person 1111161171159459134; 24.10.2015

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

Кажется, это работает на PostgreSQL 9.5

person sigod    schedule 07.07.2016
comment
Насколько я понял, это работает для меня, это удаляет поле a из данных, а затем добавляет поле a с новым значением. В моем случае значение a было основано на столбце. ОБНОВЛЕНИЕ test SET data = data :: jsonb - 'a' || ('{a:' || myColumn || '}') :: jsonb; - person sebge2; 15.02.2018

Вы можете попробовать обновить, как показано ниже:

Синтаксис: UPDATE имя_таблицы SET имя_столбца = имя_столбца :: jsonb || '{key: new_value}' WHERE имя_столбца условие;

Для вашего примера:

ОБНОВЛЕНИЕ test SET data = data :: jsonb || '{a: new_value}' ГДЕ данные - ›› 'b' = '2';

person Ganesh Bhise    schedule 03.07.2020

Это сработало для меня при попытке обновить поле строкового типа.

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

Надеюсь, это поможет кому-то другому!

Предполагая, что таблица table_name имеет столбец jsonb с именем body, и вы хотите изменить body.some_key = 'value'

person Antonio    schedule 09.03.2018
comment
к сожалению, это переформатирует JSON так же, как и манипуляции с помощью специфичных для JSON функций. - person Lu55; 27.03.2018

В 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 дней, поэтому, если вам нужно, чтобы он работал в устаревших системах и у вас есть свободное время, это может стоить усилий. Я полагаю, что версия C намного быстрее, чем версия на Python.

person Magnus    schedule 24.02.2015

Даже если следующее не удовлетворит этот запрос (функция 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 );
person Ezequiel Tolnay    schedule 25.11.2015

Я написал для себя небольшую функцию, которая рекурсивно работает в 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)

Как видите, он глубоко анализирует и обновляет / добавляет значения там, где это необходимо.

person J. Raczkiewicz    schedule 13.10.2016

Я обнаружил, что предыдущие ответы подходят опытным пользователям 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», ведущий к (и включая) ключ, который мы собираемся перезаписать.

"новое_значение" - это новое значение, которое мы присваиваем

в нашем случае мы хотим обновить значение key13, которое находится под key1 (key1 -> key13):

отсюда синтаксис пути: '{key1, key13}' (путь был самым сложным для взлома - потому что учебники ужасны)

jsonb_set(jsonb_column,'{key1,key13}','"See main screen labeling"')
person Dror    schedule 20.06.2019

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

Например, в 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';
person Roman Pekar    schedule 24.08.2013
comment
Жаль, что Amazon RDS не поддерживает plpython3u! - person dbau; 23.02.2014
comment
value также потребует loads при установке нечисловых значений, таких как строки (js[key] = loads(value)) - В противном случае: select json_update('{"a":"a"}', 'a', to_json('b')); -> {"a": "\"b\""} - person hooblei; 12.05.2014
comment
Этот ответ также можно изменить, чтобы включить удаление ключа, когда значение установлено на None: `if value is None: del data [key] - person Joshua Burns; 23.03.2016

Следующий фрагмент кода 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';
person Sandeep    schedule 29.07.2015

Вы также можете атомарно увеличивать ключи внутри 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

person joonas.fi    schedule 22.08.2016

что вы думаете об этом решении?

Он добавит новое значение или обновит существующее.

Изменить: отредактировано, чтобы заставить его работать с нулевым и пустым объектом

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)
person Haerphi    schedule 13.08.2020

Для тех, кто использует 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,
person user218867    schedule 15.08.2019

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

Это то, что сработало для меня, attrs - это поле типа json. сначала приведите к jsonb, затем обновите.

or

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

person pooj17    schedule 29.07.2020

Если вы хотите использовать значения из других столбцов в своей команде обновления JSON, вы можете использовать конкатенацию строк:

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

Так, например, моя строка выглядит так: {a1: {a11: x, a22: y, a33: z}}

Я обновляю 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;

Это больше связано со строкой, чем с json, но работает. По сути, он извлекает все данные во временную таблицу, создает строку, закрывая дыры concat с данными, которые вы зарезервировали, и преобразует ее в jsonb.

Json_set может быть более эффективным, но я все еще разбираюсь в нем. В первый раз, когда я попробовал его использовать, я полностью испортил струну ...

person Vlad S    schedule 21.01.2020
comment
привет и добро пожаловать в StackOverflow! Обратите внимание, что на этот вопрос уже есть принятый ответ. - person hongsy; 21.01.2020