PostgreSQL
 sql >> Datenbank >  >> RDS >> PostgreSQL

Fügen Sie mit Postgres Daten in 3 Tabellen gleichzeitig ein

Verwenden Sie datenmodifizierende CTEs :

WITH ins1 AS (
   INSERT INTO sample(firstname, lastname)
   VALUES ('fai55', 'shaggk')
-- ON     CONFLICT DO NOTHING         -- optional addition in Postgres 9.5+
   RETURNING id AS sample_id
   )
, ins2 AS (
   INSERT INTO sample1 (sample_id, adddetails)
   SELECT sample_id, 'ss' FROM ins1
   RETURNING user_id
   )
INSERT INTO sample2 (user_id, value)
SELECT user_id, 'ss2' FROM ins2;

Jedes INSERT hängt von dem davor ab. SELECT statt VALUES stellt sicher, dass nichts in untergeordnete Tabellen eingefügt wird, wenn keine Zeile von einem vorherigen INSERT zurückgegeben wird . (Seit Postgres 9.5+ können Sie einen ON CONFLICT hinzufügen .)
So ist es auch etwas kürzer und schneller.

Normalerweise ist es bequemer, vollständige Datenzeilen an einem Ort bereitzustellen :

WITH data(firstname, lastname, adddetails, value) AS (
   VALUES                              -- provide data here
      ('fai55', 'shaggk', 'ss', 'ss2') -- see below
    , ('fai56', 'XXaggk', 'xx', 'xx2') -- works for multiple input rows
       --  more?                      
   )
, ins1 AS (
   INSERT INTO sample (firstname, lastname)
   SELECT firstname, lastname          -- DISTINCT? see below
   FROM   data
   -- ON     CONFLICT DO NOTHING       -- UNIQUE constraint? see below
   RETURNING firstname, lastname, id AS sample_id
   )
, ins2 AS (
   INSERT INTO sample1 (sample_id, adddetails)
   SELECT ins1.sample_id, d.adddetails
   FROM   data d
   JOIN   ins1 USING (firstname, lastname)
   RETURNING sample_id, user_id
   )
INSERT INTO sample2 (user_id, value)
SELECT ins2.user_id, d.value
FROM   data d
JOIN   ins1 USING (firstname, lastname)
JOIN   ins2 USING (sample_id);

db<>hier fummeln

Möglicherweise benötigen Sie explizite Typumwandlungen in einem eigenständigen VALUES Ausdruck - im Gegensatz zu einem VALUES Ausdruck, der an INSERT angehängt ist wobei Datentypen aus der Zieltabelle abgeleitet werden. Siehe:

  • Umwandeln des NULL-Typs beim Aktualisieren mehrerer Zeilen

Wenn mehrere Zeilen mit identischem (firstname, lastname) kommen können , müssen Sie möglicherweise Duplikate für das erste INSERT falten :

...
INSERT INTO sample (firstname, lastname)
SELECT DISTINCT firstname, lastname FROM data
...

Anstelle der CTE-data können Sie auch eine (temporäre) Tabelle als Datenquelle verwenden .

Es wäre wahrscheinlich sinnvoll, dies mit einer UNIQUE-Einschränkung für (firstname, lastname) zu kombinieren in der Tabelle und ein ON CONFLICT -Klausel in der Abfrage.

Verwandte:

  • Wie verwendet man RETURNING mit ON CONFLICT in PostgreSQL?
  • Ist SELECT oder INSERT in einer Funktion anfällig für Race-Conditions?