情景:
假设我有两个表,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])
我问的原因是在我看来,在处理较大的牌桌时,第一种选择会更有效。
谢谢!
答案 0 :(得分:698)
DELETE TableA
FROM TableA a
INNER JOIN TableB b
ON b.Bid = a.Bid
AND [my filter condition]
应该有效
答案 1 :(得分:254)
我会使用这种语法
Delete a
from TableA a
Inner Join TableB b
on a.BId = b.BId
WHERE [filter condition]
答案 2 :(得分:29)
是的,你可以。示例:
DELETE TableA
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
答案 3 :(得分:10)
尝试使用访问数据库执行此操作,发现我需要在删除后立即使用 a。* 。
DELETE a.*
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
答案 4 :(得分:8)
在 MySQL 中几乎相同,但您必须在“DELETE”后面使用表别名:
DELETE a
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
答案 5 :(得分: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关键字)
答案 6 :(得分: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())
如果你可以从这个脚本中得到充分的理由或写下另一个有用的,请分享。谢谢,希望得到这个帮助。
答案 7 :(得分:1)
假设您有2个表,一个具有Master集(例如Employees),另一个具有子集(例如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表上执行删除操作。
答案 8 :(得分:1)
在SQLite中,唯一可行的是类似于beauXjames的回答。
似乎归结为此
DELETE FROM table1 WHERE table1.col1 IN (SOME TEMPORARY TABLE);
并且可以通过SELECT和JOIN来创建一些临时表,这两个表可以根据要删除Table1中记录的条件来过滤此临时表。
答案 9 :(得分:1)
您可以运行此查询: -
Delete from TableA
from
TableA a, TableB b
where a.Bid=b.Bid
AND [my filter condition]
答案 10 :(得分:0)
更简单的方法是:
DELETE TableA
FROM TableB
WHERE TableA.ID = TableB.ID
答案 11 :(得分:0)
DELETE FROM table1
where id IN
(SELECT id FROM table2..INNER JOIN..INNER JOIN WHERE etc)
使用Join最小化DML查询。您应该能够使用上述子查询来执行所有DML查询中的大多数操作。
通常,仅当需要在1+个表中的列进行SELECT或GROUP BY时,才应使用联接。如果您仅仅触摸多个表来定义总体,请使用子查询。对于DELETE查询,请使用相关子查询。