web-dev-qa-db-ja.com

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

シナリオ:

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])

私が尋ねる理由は、大きなテーブルを扱うときに最初のオプションがはるかに効果的になるだろうということです。

ありがとうございます。

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

うまくいくはず

697
TheTXI

私はこの構文を使います

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

はい、できます。例:

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

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

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

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

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

私はこれを使っています

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

@TheTXIの方法で十分ですが、私は答えとコメントを読んで、私は答えなければならない1つの事柄が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())

あなたがこのスクリプトから正当な理由を得ることができるか、または他の役に立つものを書くことができるならば、共有してください。ありがとう、そしてこれがお役に立てば幸いです。

2
QMaster

上記の構文は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
DavidJ

あなたはこのクエリを実行することができます: -

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

2つのテーブルがあり、1つはマスターセット(例:Employees)、もう1つは子セット(例、Dependents)を持ち、キーアップできないDependentsテーブルのすべてのデータ行を削除したいとします。マスターテーブル内の任意の行で。

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
beauXjames

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

このDELETE FROM table1 WHERE table1.col1 IN (SOME TEMPORARY TABLE);に辿り着くように思われ、いくつかの一時テーブルはSELECTで作成でき、あなたの2つのテーブルを結合することができます。

1

もっと簡単な方法は:

DELETE TableA
FROM TableB
WHERE TableA.ID = TableB.ID
0
Carlos Barini