ฉันจะแก้ไขฟิลด์ภายในประเภทข้อมูล 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; casts จำเป็นต้องใช้เพื่อจัดการjsonค่า)

การรวมวัตถุ JSON 2 (หรือมากกว่า) เข้าด้วยกัน (หรือเชื่อมต่ออาร์เรย์):

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()) เมื่อ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()โดยตรงกับ

คำตอบเดิม : เป็นไปได้ (โดยไม่มี plpython หรือ plv8) ใน pure 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) ค่า leaf (และใช้ฟังก์ชันแรกจากคำตอบนี้) ซึ่งอยู่ที่ key-path (ซึ่งคีย์สามารถอ้างถึงวัตถุภายในเท่านั้นไม่สนับสนุนอาร์เรย์ภายใน):

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
การดำเนินการนี้เทียบเท่านั้นไม่ใช่UPSERT UPDATEหากคีย์ยังไม่มีอยู่ในฟิลด์ json คีย์จะถูกเพิ่ม ลองดูคำถามที่เกี่ยวข้องกับเรื่องนี้จริง ๆUPDATE: stackoverflow.com/questions/7711432/… (สำหรับประเภทคอมโพสิต แต่ตัวการคล้ายกับ json)
Erwin Brandstetter

1
@ErwinBrandstetter ที่เป็นจริง แต่ใน json UPSERT มักจะกว้างกว่าการแก้ไขแบบ UPDATE (พิจารณาf.ex. sqlfiddle.com/#!15/d41d8/2897 ) - ฉันตีความคำถามเดิมว่าคุณแก้ไขอย่างไร พวกเขา (คอลัมน์ json) โดยใช้คำสั่ง UPDATE? - นอกจากเงื่อนไขเดียวสามารถเปลี่ยนเป็น UPDATE ได้
pozs

1
มีประโยชน์มากและเสร็จสมบูรณ์ในขณะนี้
Erwin Brandstetter

1
@maxhud ที่ขึ้นอยู่กับไคลเอนต์ (หรือไลบรารีไคลเอ็นต์ที่คุณใช้) หากคุณสามารถทำได้ให้ใช้ประเภทที่ชัดเจน (PostgreSQL สามารถเดาประเภทในการสืบค้นแบบ parametrized ได้ แต่โดยปกติแล้วมันจะทำงานได้ไม่ดีกับฟังก์ชั่น polymorphic) $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;มันไม่รู้จักหรือฉันจะได้รับพฤติกรรมเดียวกันได้อย่างไร ขอบคุณ
Rafael Capucho

1
หากค่าที่ฉันต้องการตั้งค่าเป็นซับสตริงจากคอลัมน์อื่นแทนที่จะเป็น "แมรี่" ฉันจะทำอย่างไร
Andrew

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)

หากต้องการอัปเดตฟิลด์ภายในอย่าใช้ตัวดำเนินการเชื่อม||ต่อ ใช้ 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ฟังก์ชัน


และถ้าฉันต้องอัปเดตสองฟิลด์แล้วไวยากรณ์คืออะไร?
Sungarg

ถ้าฉันมีคอลัมน์ 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$;

Recursive Delete By Key:ลบค่าจากโครงสร้าง 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" ต่อท้ายด้วยค่าใหม่ ในกรณีของฉันค่าของ "a" ขึ้นอยู่กับคอลัมน์ การทดสอบ UPDATE SET data = 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 หากคุณไม่กังวลเกี่ยวกับการจัดเรียงคำสั่งซื้ออสังหาริมทรัพย์ใหม่
Karthik Sivaraj

4

ด้วย PostgreSQL 9.4 เราได้ใช้ฟังก์ชั่นหลามต่อไปนี้ มันอาจทำงานกับ 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 เร็วกว่ารุ่นไพ ธ อนมาก


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ยังจะต้องมีloadsเมื่อตั้งค่าตัวเลขที่ไม่เหมือนสตริง ( js[key] = loads(value)) - มิฉะนั้น:select json_update('{"a":"a"}', 'a', to_json('b')); -> {"a": "\"b\""}
hooblei

คำตอบนี้ยังสามารถแก้ไขเพื่อรวมการลบคีย์เมื่อตั้งค่าเป็น None: `ถ้าค่าเป็น 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])

ใน " target " - ฉันจะวางชื่อคอลัมน์ jsonb (นี่คือคอลัมน์ตารางที่กำลังแก้ไข)

" path " - คือ "json keys path" ที่นำไปสู่ ​​(และรวมถึง) คีย์ที่เรากำลังจะเขียนทับ

" 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 ตัวอย่าง:

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


params:

  • 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 แต่ใช้งานได้ โดยทั่วไปมันจะดึงข้อมูลทั้งหมดลงในตาราง temp สร้างสตริงในขณะที่เสียบรู concat กับข้อมูลที่คุณสำรองและแปลงเป็น jsonb

Json_set อาจมีประสิทธิภาพมากกว่า แต่ฉันยังคงติดใจอยู่ ครั้งแรกที่ฉันพยายามใช้มันฉันทำสายให้ยุ่ง ...


1
สวัสดีและยินดีต้อนรับสู่ StackOverflow! โปรดทราบว่ามีคำตอบสำหรับคำถามนี้แล้ว
hongsy

-2

หากคุณกำลังทำแบบสอบถามนี้ด้วยไคลเอนต์ภาษาการเขียนโปรแกรมเช่นจากpython pycopg2หรือNode Postgresตรวจสอบให้แน่ใจว่าคุณได้วิเคราะห์ข้อมูลใหม่ไปยัง JSON ก่อน

มันอาจดูเหมือนพจนานุกรมงูใหญ่เหมือนกันกับวัตถุ 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()

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.