programing

새 Postgre 내의 필드를 수정하려면 어떻게 해야 합니까?SQL JSON 데이터형?

abcjava 2023. 2. 20. 23:48
반응형

새 Postgre 내의 필드를 수정하려면 어떻게 해야 합니까?SQL JSON 데이터형?

9.에서는 postgresql 9.3을 실행할 수 .SELECT타입의 입니다만, JSON 을 사용해 그것들을 하려면 , 「JSON」, 「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...

업데이트: Postgre 포함SQL 9.5, 몇 가지jsonb 에서의 (SQL자체에서는 없음)(SQL」( 「」, 「」)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 합니다.「 」 、 「 」 。

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)

라고 하면, 음의 라고 할 수 있습니다.pathJSON 어레이의 끝부터 카운트합니다.

예를 들어 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()이(가) 동작합니다.pathtargetJSON을 사용하다.이 경우 키를 사용하지 않을 때만 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$;

하드웨어

편집:

여러 키와 값을 설정하는 버전:

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;

업데이트: 함수가 압축되었습니다.

9.5에서는 jsonb_set-를 사용합니다.

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

여기서 body는 jsonb 컬럼유형입니다.

Postgresql 9.5에서는 다음과 같이 실행할 수 있습니다.

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

또는

UPDATE test
SET data = jsonb_set(data, '{a}', '5'::jsonb);

jsonb 값에서 여러 필드를 한 번에 업데이트하는 방법을 물어보셨습니다.테이블을 작성한다고 가정합니다.

CREATE TABLE testjsonb ( id SERIAL PRIMARY KEY, object JSONB );

그런 다음 실험 행을 삽입합니다.

INSERT INTO testjsonb
VALUES (DEFAULT, '{"a":"one", "b":"two", "c":{"c1":"see1","c2":"see2","c3":"see3"}}');

그런 다음 행을 업데이트합니다.

UPDATE testjsonb SET object = object - 'b' || '{"a":1,"d":4}';

그 결과, 다음과 같은 것이 실현됩니다.

  1. 필드를 업데이트합니다.
  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)

연산자 "'t the concat" 를 사용합니다.|| 을 사용합니다 【jsonb_set】【jsonb_set】【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 함수 설명서에서 파워를 구하십시오.관심 있는 사람은#- 환입니다 operator operator operator.jsonb_set 및 "function"도 마찬가지입니다.jsonb_insert★★★★★★ 。

필드 유형이 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 함수연산자

여기서 제 노트를 읽을 수 있습니다.

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

Postgre에서 효과가 있는 것 같습니다.SQL 9.5

다음과 같이 업데이트를 시도할 수 있습니다.

구문: UPDATE table_name SET column_name = column_name::jsonb | '{key":new_value}' Where column_name 조건;

예:

테스트 세트 데이터 업데이트 = 데이터::jsonb || '{"a":new_value}' 여기서 data->'b' = '2';

문자열 유형 필드를 업데이트하려고 할 때 이 방법이 효과가 있었습니다.

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

다른 사람에게 도움이 되었으면 좋겠네요!

table_name 테이블에 body라는 이름의 jsonb 열이 있고 body를 변경하는 경우.some_key = '값'

나는 이전의 답변이 경험이 많은 포스트그레에게 더 적합하다는 것을 알았다.SQL 사용자이것은 초보자용입니다.

다음 값을 가진 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() 함수를 사용하여 새로운 값을 키에 할당합니다.

jsonb_set()에 대한 파라미터

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

"target" - jsonb column-name을 배치합니다(이것이 수정되는 테이블 열입니다).

"path"는 덮어쓰는 키로 이어지는 (그리고 포함) "json keys path"입니다.

"new_value" - 이것이 새로운 값입니다.

이 경우 key1(key1 -> key13) 아래에 있는 key13의 값을 갱신합니다.

따라서 경로 구문은 '{key1,key13}'입니다(자습서가 형편없기 때문에 경로가 크래킹하기 가장 까다로운 부분이었습니다).

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

@poz의 답변을 바탕으로 Postgre를 몇 개 더 소개합니다.일부 사용자에게 유용한 SQL 함수(Postgre 필요)SQL 9.3+)

Delete By Key : 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"}}

포스트그레 이후SQL 14에서는 jsonb Subscripting을 사용하여 JSON 필드의 요소에 직접 액세스하여 최종적으로 업데이트할 수 있습니다.

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

Postgre 포함SQL 9.4에서는 다음과 같은 python 함수를 구현했습니다.Postgre에서도 동작할 수 있습니다.SQL 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;

고용주의 JSON 하기 위한 에 주의해 주십시오(은 아닙니다).json ★★★★★★★★★★★★★★★★★」jsonb (Postgrepostgre 우))타입)SQL 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형입니다.

다음으로는 이 요구를 충족할 수 없습니다(json_object_agg 함수는 Postgre에서는 사용할 수 없습니다).SQL 9.3)은 Postgre용 || 오퍼레이터를 찾고 있는 모든 사용자에게 도움이 됩니다.SQL 9.4 (향후 Postgre에서 구현)SQL 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 );

atomically 에서는 키를 수도 .jsonb음음음같 뭇매하다

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

SELECT * FROM users;

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

정의되지 않은 키 ->는 시작 값을 0으로 가정합니다.

상세한 것에 대하여는, https://stackoverflow.com/a/39076637 를 참조해 주세요.

Postgres 9.4에서 재귀적으로 작동하는 작은 함수를 직접 작성했습니다.이 기능은 다음과 같습니다(잘 되길 바랍니다).

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

    result = val1;

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

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

    RETURN result;
END;
$$ LANGUAGE plpgsql;

다음은 사용 예를 제시하겠습니다.

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

보시는 바와 같이 깊이 분석하여 필요에 따라 값을 업데이트/추가합니다.

문서에서는 아무것도 찾을 수 없습니다만, 예를 들어 확장 함수를 작성하는 등의 회피책을 사용할 수 있습니다.

예를 들어 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';

다음 플피톤 조각이 유용할 수 있습니다.

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

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

이게 나한테 효과가 있었어. atts는 json 타입의 필드야.먼저 jsonb에 캐스팅한 후 업데이트합니다.

또는

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

당신은 이 해결책에 대해 어떻게 생각하십니까?

새 값을 추가하거나 기존 값을 업데이트합니다.

편집: null 및 빈 개체와 함께 작동하도록 편집됨

Edit2: 객체의 개체와 함께 작동하도록 편집...

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

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

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

사용방법:

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

하시는 분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는 필드값의★★★★★★★★★★★★★★★★★★★★★★★·
    들어 됩니다.jackson ,

JSON update 명령에서 다른 열의 값을 사용하는 경우 문자열 연결을 사용할 수 있습니다.

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

예를 들어 문자열은 다음과 같습니다.{ " a1 " : { " a11 " : " x " , a22 " " " " y " , a33 " " : z " }

temp table을 사용하여 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보다는 string과 더 관련이 있지만 효과가 있습니다.기본적으로 모든 데이터를 임시 테이블로 가져와서 백업한 데이터로 콘캣 구멍을 막으면서 문자열을 생성하여 jsonb로 변환합니다.

Json_set이 더 효율적일 수 있지만, 아직 감이 잡히지 않았습니다.처음 사용하려고 했을 때 끈이 완전히 엉망이 되어 버려서...

새 필드도 추가할 수 있습니다.

타이포름 코드

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

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

이 솔루션은 다음 중 하나입니다.jsonb_setJSONB는 NULL로 되어 있습니다.jsonb_set는 개체가 존재하는 경우에만 작동합니다.

다음 예시에서 설정은 사용자 테이블의 JSONB 열입니다.

 UPDATE public."Users"
 SET settings = coalesce("settings", '{}')::jsonb || '{ "KeyToSet" : "ValueToSet" }'
 WHERE id=35877;
select * from pg_settings where name = 'deadlock_timeout';
begin;
create temp table a2(data jsonb);
insert into a2 values('{
    "key0": {
        "key01": "2018-05-06T12:36:11.916761+00:00",
        "key02": "DEFAULT_WEB_CONFIGURATION",
    "key1": {
        "key11": "Data System",
        "key12": "<p>Health,<p>my address<p>USA",
        "key13": "*Please refer to main screen labeling"
    }
}}'::jsonb);
commit;

중첩된 jsonb 구조 업데이트.삭제에 적용할 수 있습니다.

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

언급URL : https://stackoverflow.com/questions/18209625/how-do-i-modify-fields-inside-the-new-postgresql-json-datatype

반응형