ChatGPT解决这个技术问题 Extra ChatGPT

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

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

谢谢!


P
Pரதீப்
DELETE TableA
FROM   TableA a
       INNER JOIN TableB b
               ON b.Bid = a.Bid
                  AND [my filter condition] 

应该管用


我在连接上使用了 And [my filter condition] 而不是 Where 子句。我想两者都可以工作,但加入的过滤条件会限制你的加入结果。
一个问题。为什么我们需要写“DELETE TableA FROM”而不是“DELETE FROM”?我看到它只在这种情况下有效,但为什么呢?
我认为是因为您必须指出要从哪个表中删除记录。我刚刚使用语法 DELETE TableA, TableB ... 运行了一个查询,实际上从两者中删除了相关记录。好的。
在 PostgreSQL 中,带有 join 的语法不起作用,但可以使用“using”关键字。 DELETE from TableA a using TableB b where b.Bid = a.Bid and [my filter condition]
在 MySQL 中,您会收到错误“MULTI DELETE 中的未知表 'TableA'”,这是因为您为 TableA (a) 声明了别名。小调整:DELETE a FROM TableA a INNER JOIN TableB b on b.Bid = a.Bid and [my filter condition]
c
cmsjr

我会使用这种语法

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

我也更喜欢这种语法,似乎在逻辑上更有意义。另外,我知道您可以对 UPDATE 使用相同类型的语法。
我也更喜欢它,因为在 DELETE 之后放置表别名对我来说总是更直观地了解要删除的内容。
事实上,这对我来说也是首选。特别是在我需要实际加入同一张表的情况下(例如删除重复记录)。在这种情况下,我需要为要从中删除的“边”使用别名,并且这种语法非常清楚我正在从重复别名中删除。
D
Diadistis

是的你可以。例子 :

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

我更喜欢通过别名来引用第一行中的表格。那是“删除a”而不是“删除TableA”。在您将表与自身连接的情况下,它会明确您要删除哪一侧。
T
Tony Emrud

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

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

来自被拒绝的待编辑:“必须将 UniqueRecords 属性设置为是,否则它将不起作用。(support.microsoft.com/kb/240098)”
M
Michael Butler

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

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

D
DavidJ

上面的语法在 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 关键字)


Q
QMaster

我正在使用这个

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

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


b
beauXjames

假设您有 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 表执行删除操作。


B
Bhoom Suktitipat

在 SQLite 中,唯一有效的是类似于 beauXjames 的回答。

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


D
Druid

您可以运行此查询:

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

B
ByteHamster

更简单的方法是:

DELETE TableA
FROM TableB
WHERE TableA.ID = TableB.ID

1
1c1cle
DELETE FROM table1
where id IN 
    (SELECT id FROM table2..INNER JOIN..INNER JOIN WHERE etc)

尽量减少对连接的 DML 查询的使用。您应该能够使用上述子查询执行大多数 DML 查询。

通常,仅当您需要按 2 个或更多表中的列进行 SELECT 或 GROUP 时才应使用连接。如果您只是接触多个表来定义总体,请使用子查询。对于 DELETE 查询,使用相关子查询。