1つのクエリで3つのテーブルにデータを挿入したい。
私のテーブルは次のようになります。
CREATE TABLE sample (
id bigserial PRIMARY KEY,
lastname varchar(20),
firstname varchar(20)
);
CREATE TABLE sample1(
user_id bigserial PRIMARY KEY,
sample_id bigint REFERENCES sample,
adddetails varchar(20)
);
CREATE TABLE sample2(
id bigserial PRIMARY KEY,
user_id bigint REFERENCES sample1,
value varchar(10)
);
挿入ごとにキーを取得し、そのキーを次のテーブルに挿入する必要があります。
私のクエリは:
insert into sample(firstname,lastname) values('fai55','shaggk') RETURNING id;
insert into sample1(sample_id, adddetails) values($id,'ss') RETURNING user_id;
insert into sample2(user_id, value) values($id,'ss') RETURNING id;
しかし、単一のクエリを実行すると、値が返されるだけで、次のクエリですぐに再利用することはできません。
これを達成する方法は?
data-modifying 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 user_id
)
, ins2 AS (
INSERT INTO sample1 (user_id, adddetails)
SELECT user_id, 'ss' FROM ins1
-- RETURNING user_id -- only if used in turn
)
INSERT INTO sample2 (user_id, value) -- same here
SELECT user_id, 'ss' FROM ins1;
各INSERT
は前のものに依存します。 SELECT
の代わりにVALUES
は、前のINSERT
から行が返されない場合、補助テーブルに何も挿入されないようにします。 (関連:Postgres 9.5+の ON CONFLICT
句)
この方法でも少し短く、高速です。
通常、1つの場所に完全なデータ行を提供する方が便利です:
WITH data(firstname, lastname, adddetails, value) AS (
VALUES -- provide data here
(text 'fai55', text 'shaggk', text 'ss', text 'ss2') -- see below
-- more? -- works for multiple input rows
)
, ins1 AS (
INSERT INTO sample (firstname, lastname)
SELECT firstname, lastname FROM data -- DISTINCT? see below
ON CONFLICT DO NOTHING -- required UNIQUE constraint
RETURNING firstname, lastname, id AS sample_id
)
, ins2 AS (
INSERT INTO sample1 (sample_id, adddetails)
SELECT sample_id, adddetails
FROM data
JOIN ins1 USING (firstname, lastname)
RETURNING sample_id, user_id
)
INSERT INTO sample2 (user_id, value)
SELECT user_id, value
FROM data
JOIN ins1 USING (firstname, lastname)
JOIN ins2 USING (sample_id);
個別のVALUES
式で明示的な型キャストが必要になる場合があります(データ型がターゲットテーブルから派生するINSERTにアタッチされたVALUES
式とは異なります)。
複数の行に同一の(firstname, lastname)
が含まれる場合、最初の挿入で重複を折りたたむ必要がある場合があります。
...
INSERT INTO sample (firstname, lastname)
SELECT DISTINCT firstname, lastname FROM data
...
CTE data
の代わりに(一時)テーブルをデータソースとして使用できます。
関連、詳細:
このようなもの
with first_insert as (
insert into sample(firstname,lastname)
values('fai55','shaggk')
RETURNING id
),
second_insert as (
insert into sample1( id ,adddetails)
values
( (select id from first_insert), 'ss')
RETURNING user_id
)
insert into sample2 ( id ,adddetails)
values
( (select user_id from first_insert), 'ss');
sample2
への挿入から生成されたIDは必要ないため、最後の挿入からreturning
句を削除しました。
通常、複雑なクエリの作成を避けるためにトランザクションを使用します。
http://www.postgresql.org/docs/current/static/sql-begin.html
http://dev.mysql.com/doc/refman/5.7/en/commit.html
Postgresタグが正しいと仮定して、CTEを使用することもできます。例えば:
with sample_ids as (
insert into sample(firstname, lastname)
values('fai55','shaggk')
RETURNING id
), sample1_ids as (
insert into sample1(id, adddetails)
select id,'ss'
from sample_ids
RETURNING id, user_id
)
insert into sample2(id, user_id, value)
select id, user_id, 'val'
from sample1_ids
RETURNING id, user_id;
Sampleテーブルにafter insertトリガーを作成して、他の2つのテーブルに挿入できます。
私がこれを行うことで見られる唯一の問題は、adddetailsを挿入する方法がないことです。adddetailsは常に空またはこの場合はssです。サンプルテーブルに実際にはないサンプルに列を挿入する方法はないため、最初の挿入と一緒に送信することはできません。
別のオプションは、挿入を実行するストアドプロシージャを作成することです。
Mysqlとpostgressqlというタグが付いた質問があります。ここでどのデータベースについて話しているのですか?