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的答案。
它似乎归结为这一点
并且可以通过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 |