519

シナリオ:

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

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

ありがとう!

4

12 に答える 12

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

動作するはずです

于 2009-01-13T16:50:12.323 に答える
268

この構文を使用します

Delete a 
from TableA a
Inner Join TableB b
on  a.BId = b.BId
WHERE [filter condition]
于 2009-01-13T16:51:51.010 に答える
30

はい、できます。例 :

DELETE TableA 
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
于 2009-01-13T16:50:18.377 に答える
11

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

DELETE a.*
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
于 2011-07-22T12:55:07.263 に答える
9

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

DELETE a
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
于 2010-08-02T14:58:11.753 に答える
3

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

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キーワードをサポートしていないことに注意してください)

于 2011-11-19T19:14:58.847 に答える
3

私はこれを使用しています

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

このスクリプトから十分な理由を得ることができるか、別の有用なスクリプトを作成できる場合は、共有してください。ありがとう、そしてこの助けを願っています。

于 2014-12-22T17:19:44.530 に答える
2

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 テーブルで削除操作を実行していることです。

于 2012-06-13T18:36:35.627 に答える
2

SQLite で機能するのは、beauXjames の回答に似たものだけです。

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

于 2012-07-20T17:31:01.920 に答える
2

次のクエリを実行できます。

    DELETE FROM TableA
    FROM
       TableA a, TableB b 
    WHERE
       a.Bid=b.Bid
    AND
       [my filter condition]
于 2014-01-10T13:27:53.683 に答える