关于tsql:T-SQL:选择要通过连接删除的行

T-SQL: Selecting rows to delete via joins

场景:

假设我有两个表,TableA和TableB。 TableB的主键是单列(BId),是TableA中的外键列。

在我的情况下,我想删除TableA中与TableB中的特定行链接的所有行:我可以通过连接执行此操作吗? 删除从连接中提取的所有行?

1
2
3
4
5
6
DELETE FROM TableA
FROM
   TableA a
   INNER JOIN TableB b
      ON b.BId = a.BId
      AND [my filter condition]

或者我被迫这样做:

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

我问的原因是在我看来,在处理较大的表时,第一个选项会更有效。

谢谢!


1
2
3
4
5
DELETE TableA
FROM   TableA a
       INNER JOIN TableB b
               ON b.Bid = a.Bid
                  AND [my filter condition]

应该管用


我会使用这种语法

1
2
3
4
5
Delete a
from TableA a
Inner Join TableB b
on  a.BId = b.BId
WHERE [filter condition]


是的你可以。示例:

1
2
3
4
5
DELETE TableA
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]


尝试使用访问数据库执行此操作,发现我需要在删除后立即使用。*。

1
2
3
4
5
DELETE a.*
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]


它在MySQL中几乎相同,但您必须在"DELETE"后面使用表别名:

1
2
3
4
5
DELETE a
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]

我正在使用它

1
2
3
4
5
DELETE TableA
FROM TableA a
INNER JOIN
TableB b on b.Bid = a.Bid
AND [condition]

和@TheTXI的方式很好,但我读了答案和评论,我发现必须回答的一件事是在WHERE子句中使用条件或作为连接条件。所以我决定测试它并写一个片段,但没有发现它们之间有意义的区别。你可以在这里看到sql脚本,重要的一点是我更喜欢把它写成commnet,因为这不是确切的答案,但它很大,不能放在评论中,请原谅我。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
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())

如果你可以从这个脚本中得到充分的理由或写下另一个有用的,请分享。谢谢,希望得到这个帮助。


上面的语法在Interbase 2007中不起作用。相反,我不得不使用类似的东西:

1
2
3
4
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关键字)


你可以运行这个查询: -

1
2
3
4
5
Delete from TableA
from
TableA a, TableB b
where a.Bid=b.Bid
AND [my filter condition]

在SQLite中,唯一有效的东西类似于beauXjames的答案。

它似乎归结为这一点
DELETE FROM table1 WHERE table1.col1 IN (SOME TEMPORARY TABLE);
并且可以通过SELECT和JOIN来创建一些临时表,您可以根据要删除Table1中的记录的条件来过滤此临时表。


假设您有2个表,一个具有Master集(例如Employees),另一个具有子集(例如Dependents),并且您想要删除Dependents表中无法键入的所有数据行与主表中的任何行。

1
2
3
4
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
2
3
DELETE TableA
FROM TableB
WHERE TableA.ID = TableB.ID