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

场景:

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

在我的情况下,我想删除TableA中与TableB中特定行链接的所有行:我可以通过连接来做到这一点吗?删除从连接中拉入的所有行?

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

我问这个问题的原因是,在处理较大的表时,第一种选择似乎更有效。

谢谢!

430411 次浏览
DELETE TableA
FROM   TableA a
INNER JOIN TableB b
ON b.Bid = a.Bid
AND [my filter condition]

应该工作

是的,你可以。例子:

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

我会使用这种语法

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

它在MySQL中几乎相同,但你必须在单词“DELETE”之后使用表别名:

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

试图用访问数据库做到这一点,发现我需要在删除后立即使用一。*

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

上面的语法在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个表,其中一个具有Master集(例如。员工)和一个有孩子的集合(例如。Dependents),您希望摆脱Dependents表中所有不能与Master表中的任何行进行键合的数据行。

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集在dependencies表上执行删除操作。

在SQLite中,唯一工作的是类似于beauXjames的答案。

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

你可以运行这个查询:

    DELETE FROM TableA
FROM
TableA a, TableB b
WHERE
a.Bid=b.Bid
AND
[my filter condition]

我在用这个

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

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

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

如果你能从这个脚本中得到好的理由或者写出另一个有用的,请分享。谢谢,希望有帮助。

更简单的方法是:

DELETE TableA
FROM TableB
WHERE TableA.ID = TableB.ID
DELETE FROM table1
where id IN
(SELECT id FROM table2..INNER JOIN..INNER JOIN WHERE etc)

尽量减少使用连接的DML查询。您应该能够使用上面的子查询执行大部分DML查询。

一般来说,只有在需要在两个或多个表中按列进行SELECT或GROUP时才应该使用连接。如果你的只有触摸多个表来定义一个填充,使用子查询。对于DELETE查询,使用相关子查询。