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?