sql >> Databáze >  >> RDS >> PostgreSQL

Vkládejte data do 3 tabulek najednou pomocí Postgres

Použijte CTE upravující data :

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;

Každý INSERT záleží na předchozí. SELECT místo VALUES zajišťuje, že do vedlejších tabulek není nic vloženo, pokud není vrácen žádný řádek z předchozího INSERT . (Od Postgresu 9.5+ můžete přidat ON CONFLICT .)
Takto je to také o něco kratší a rychlejší.

Obvykle je pohodlnější poskytnout úplné datové řádky na jednom místě :

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

Možná budete potřebovat explicitní přetypování v samostatném VALUES výraz – na rozdíl od VALUES výraz připojený k INSERT kde jsou datové typy odvozeny z cílové tabulky. Viz:

  • Při aktualizaci více řádků odesílání typu NULL

Pokud může mít více řádků stejné (firstname, lastname) , možná budete muset skládat duplikáty pro první INSERT :

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

Místo data CTE můžete jako zdroj dat použít (dočasnou) tabulku .

Pravděpodobně by dávalo smysl kombinovat toto s omezením UNIQUE na (firstname, lastname) v tabulce a ON CONFLICT klauzule v dotazu.

Související:

  • Jak používat RETURNING s ON CONFLICT v PostgreSQL?
  • Je SELECT nebo INSERT ve funkci náchylný k závodům?


  1. Vyplňte rozbalovací pole z tabulky mySQL v PHP

  2. Pro Nvarchar (Max) dostávám pouze 4000 znaků v TSQL?

  3. Trace Flag 2389 a nový odhad kardinality

  4. Zálohujte databázi na SQL Server 2017