T-SQL:結合を介して削除する行の選択


494

シナリオ:

TableAとTableBの2つのテーブルがあるとします。TableBの主キーは単一列(BId)であり、TableAの外部キー列です。

私の状況では、TableBの特定の行にリンクされているTableAのすべての行を削除したいのですが、結合によって削除できますか?結合からプルされたすべての行を削除しますか?

DELETE FROM TableA 
FROM
   TableA a
   INNER JOIN TableB b
      ON b.BId = a.BId
      AND [my filter condition]

または私はこれを行うことを余儀なくされています:

DELETE FROM TableA
WHERE
   BId IN (SELECT BId FROM TableB WHERE [my filter condition])

私が尋ねる理由は、より大きなテーブルを扱う場合、最初のオプションの方がはるかに効率的であるように思えるからです。

ありがとう!

回答:


723
DELETE TableA
FROM   TableA a
       INNER JOIN TableB b
               ON b.Bid = a.Bid
                  AND [my filter condition] 

うまくいくはず


1
Where句ではなく、結合にAnd [my filter condition]を使用しました。両方が機能すると思いますが、結合のフィルター条件により、結合からの結果が制限されます。
TheTXI 2009年

10
一つの質問。「DELETE FROM」の代わりに「DELETE TableA FROM」を記述する必要があるのはなぜですか?この場合にのみ機能するようですが、なぜですか?
LaBracca

66
レコードを削除するテーブルを指定する必要があるためです。私は構文DELETE TableA, TableB ...を使用してクエリを実行しただけで、実際に両方から関連するレコードが削除されました。いいね。
Andrew

1
PostgreSQLでは、結合を使用した構文は機能しませんが、「using」キーワードを使用できます。 DELETE from TableA a using TableB b where b.Bid = a.Bid and [my filter condition]
bartolo-otrit

8
MySQLでは、「MULTI DELETEの不明なテーブル 'TableA'」というエラーが発生します。これは、TableA(a)のエイリアスを宣言したためです。微調整:DELETE a FROM TableA a INNER JOIN TableB b on b.Bid = a.Bid and [my filter condition]
masam

260

この構文を使用します

Delete a 
from TableA a
Inner Join TableB b
on  a.BId = b.BId
WHERE [filter condition]

7
私はこの構文も好きですが、何が起こっているのかを論理的にもう少し理解できるようです。また、これと同じタイプの構文をUPDATEに使用できることも知っています。
Adam Nofsinger、2010年

DELETEの後のテーブルエイリアスの配置は、何が削除されるかについて常に私にとってより直感的に思えたので、私もそれを好みます。
ヤークト

14
確かに、これは私にとっても好まれます。具体的には、同じテーブルで実際に結合する必要がある場合(重複レコードを削除する場合など)。その場合、削除する「サイド」のエイリアスを使用する必要があります。この構文により、重複するエイリアスから削除することが非常に明確になります。
Chris Simmons、

29

はい、できます。例:

DELETE TableA 
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]

8
最初の行のテーブルはエイリアスで参照することを好みます。これは、「テーブルの削除A」ではなく「削除a」です。テーブルをそれ自体と結合する場合、どちらの側を削除するかが明確になります。
ジェレミースタイン

10

これをアクセスデータベースで実行しようとして、削除直後にa。*を使用する必要があることがわかりました。

DELETE a.*
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]

拒否された保留中の編集から:「UniqueRecordsプロパティはyesに設定する必要があります。そうしないと機能しません。(support.microsoft.com/kb/240098)」
StuperUser

8

これはMySQLでもほぼ同じですが、「DELETE」という単語の直後にテーブルエイリアスを使用する必要があります。

DELETE a
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]

2

上記の構文はInterbase 2007では機能しません。代わりに、次のようなものを使用する必要がありました。

DELETE FROM TableA a WHERE [filter condition on TableA] 
  AND (a.BId IN (SELECT a.BId FROM TableB b JOIN TableA a 
                 ON a.BId = b.BId 
                 WHERE [filter condition on TableB]))

(InterbaseはエイリアスのASキーワードをサポートしていません)


2

私はこれを使っています

DELETE TableA 
FROM TableA a
INNER JOIN
TableB b on b.Bid = a.Bid
AND [condition]

@TheTXIの方法は十分ですが、回答とコメントを読んだところ、WHERE句の条件を使用するか、結合条件として回答する必要があることがわかりました。だから私はそれをテストしてスニペットを書くことにしましたが、それらの間に意味のある違いは見つかりませんでした。ここでsqlスクリプトを見ることができますが、重要な点は、これは正確な答えではないため、commnetとして記述するほうがよいということですが、サイズが大きく、コメントを入力できないため、ご容赦ください。

Declare @TableA  Table
(
  aId INT,
  aName VARCHAR(50),
  bId INT
)
Declare @TableB  Table
(
  bId INT,
  bName VARCHAR(50)  
)

Declare @TableC  Table
(
  cId INT,
  cName VARCHAR(50),
  dId INT
)
Declare @TableD  Table
(
  dId INT,
  dName VARCHAR(50)  
)

DECLARE @StartTime DATETIME;
SELECT @startTime = GETDATE();

DECLARE @i INT;

SET @i = 1;

WHILE @i < 1000000
BEGIN
  INSERT INTO @TableB VALUES(@i, 'nameB:' + CONVERT(VARCHAR, @i))
  INSERT INTO @TableA VALUES(@i+5, 'nameA:' + CONVERT(VARCHAR, @i+5), @i)

  SET @i = @i + 1;
END

SELECT @startTime = GETDATE()

DELETE a
--SELECT *
FROM @TableA a
Inner Join @TableB b
ON  a.BId = b.BId
WHERE a.aName LIKE '%5'

SELECT Duration = DATEDIFF(ms,@StartTime,GETDATE())

SET @i = 1;
WHILE @i < 1000000
BEGIN
  INSERT INTO @TableD VALUES(@i, 'nameB:' + CONVERT(VARCHAR, @i))
  INSERT INTO @TableC VALUES(@i+5, 'nameA:' + CONVERT(VARCHAR, @i+5), @i)

  SET @i = @i + 1;
END

SELECT @startTime = GETDATE()

DELETE c
--SELECT *
FROM @TableC c
Inner Join @TableD d
ON  c.DId = d.DId
AND c.cName LIKE '%5'

SELECT Duration    = DATEDIFF(ms,@StartTime,GETDATE())

このスクリプトから十分な理由が得られるか、別の有用な記述ができる場合は、共有してください。おかげで、この助けを願っています。


1

たとえば、マスターセット(例:従業員)と子セット(例:依存関係)の2つのテーブルがあり、キーアップできない依存関係テーブルのデータ行をすべて削除したいとします。マスターテーブルの行を使用します。

delete from Dependents where EmpID in (
select d.EmpID from Employees e 
    right join Dependents d on e.EmpID = d.EmpID
    where e.EmpID is null)

ここで注目すべき点は、最初に結合からEmpIDの「配列」を収集しているだけであり、そのEmpIDのセットを使用してDependentsテーブルで削除操作を実行していることです。


1

SQLiteでは、動作するのはbeauXjamesの答えに似たものだけです。

これ DELETE FROM table1 WHERE table1.col1 IN (SOME TEMPORARY TABLE); に帰着するようで、いくつかの一時テーブルはSELECTで作成でき、2つのテーブルを結合して、Table1のレコードを削除する条件に基づいてこの一時テーブルをフィルタリングできます。


1

このクエリを実行できます:-

Delete from TableA 
from 
TableA a, TableB b 
where a.Bid=b.Bid
AND [my filter condition]


1
DELETE FROM table1
where id IN 
    (SELECT id FROM table2..INNER JOIN..INNER JOIN WHERE etc)

結合でのDMLクエリの使用を最小限に抑えます。上記のようなサブクエリを使用して、ほとんどすべてのDMLクエリを実行できるはずです。

一般に、結合は、2つ以上のテーブルの列でSELECTまたはGROUPする必要がある場合にのみ使用する必要があります。母集団を定義するために複数のテーブルのみを操作する場合は、サブクエリを使用します。DELETEクエリの場合は、相関サブクエリを使用します。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.