T-SQL: выбор строк для удаления с помощью объединений - PullRequest
476 голосов
/ 13 января 2009

Сценарий:

Допустим, у меня есть две таблицы, 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])

Причина, по которой я спрашиваю, заключается в том, что мне кажется, что первый вариант был бы гораздо более эффективным при работе с большими таблицами.

Спасибо!

Ответы [ 11 ]

698 голосов
/ 13 января 2009
DELETE TableA
FROM   TableA a
       INNER JOIN TableB b
               ON b.Bid = a.Bid
                  AND [my filter condition] 

должно работать

254 голосов
/ 13 января 2009

Я бы использовал этот синтаксис

Delete a 
from TableA a
Inner Join TableB b
on  a.BId = b.BId
WHERE [filter condition]
29 голосов
/ 13 января 2009

Да, вы можете. Пример:

DELETE TableA 
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
10 голосов
/ 22 июля 2011

Пытался сделать это с базой данных доступа и обнаружил, что мне нужно использовать a. * сразу после удаления.

DELETE a.*
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
8 голосов
/ 02 августа 2010

Это почти то же самое в MySQL , но вы должны использовать псевдоним таблицы сразу после слова "УДАЛИТЬ":

DELETE a
FROM TableA AS a
INNER JOIN TableB AS b
ON a.BId = b.BId
WHERE [filter condition]
2 голосов
/ 22 декабря 2014

Я использую это

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

Если вы могли бы получить вескую причину из этого скрипта или написать другую полезную, пожалуйста, поделитесь. Спасибо и надеюсь, что это поможет.

2 голосов
/ 19 ноября 2011

Синтаксис выше не работает в 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 для псевдонимов)

1 голос
/ 10 января 2014

Вы можете выполнить этот запрос: -

Delete from TableA 
from 
TableA a, TableB b 
where a.Bid=b.Bid
AND [my filter condition]
1 голос
/ 20 июля 2012

В SQLite единственное, что работает, - это что-то похожее на ответ beauXjames.

Кажется, все сводится к этому DELETE FROM table1 WHERE table1.col1 IN (SOME TEMPORARY TABLE); и что некоторая временная таблица может быть создана с помощью SELECT и JOIN вашей двух таблиц, которые вы можете отфильтровать эту временную таблицу, исходя из условия, что вы хотите удалить записи в Table1.

1 голос
/ 13 июня 2012

Допустим, у вас есть 2 таблицы, одна с основным набором (например, сотрудники) и одна с дочерним набором (например, 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.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...