SQL - найти записи из одной таблицы, которые не существуют в другой

215

У меня есть следующие две таблицы SQL (в MySQL):

Phone_book
+----+------+--------------+
| id | name | phone_number |
+----+------+--------------+
| 1  | John | 111111111111 |
+----+------+--------------+
| 2  | Jane | 222222222222 |
+----+------+--------------+

Call
+----+------+--------------+
| id | date | phone_number |
+----+------+--------------+
| 1  | 0945 | 111111111111 |
+----+------+--------------+
| 2  | 0950 | 222222222222 |
+----+------+--------------+
| 3  | 1045 | 333333333333 |
+----+------+--------------+

Как узнать, какие вызовы были сделаны людьми, у которых phone_number нет в Phone_book? Желаемый результат:

Call
+----+------+--------------+
| id | date | phone_number |
+----+------+--------------+
| 3  | 1045 | 333333333333 |
+----+------+--------------+

Любая помощь будет высоко оценена.

Теги:

8 ответов

309
Лучший ответ

Существует несколько разных способов сделать это с различной эффективностью в зависимости от того, насколько хорош ваш оптимизатор запросов и относительный размер двух таблиц:

Это кратчайший оператор и может быть самым быстрым, если ваша телефонная книга очень короткая:

SELECT  *
FROM    Call
WHERE   phone_number NOT IN (SELECT phone_number FROM Phone_book)

в качестве альтернативы (благодаря Alterlife)

SELECT *
FROM   Call
WHERE  NOT EXISTS
  (SELECT *
   FROM   Phone_book
   WHERE  Phone_book.phone_number = Call.phone_number)

или (благодаря WOPR)

SELECT * 
FROM   Call
LEFT OUTER JOIN Phone_Book
  ON (Call.phone_number = Phone_book.phone_number)
  WHERE Phone_book.phone_number IS NULL

(игнорируя, что, как говорили другие, обычно лучше всего выбирать только нужные столбцы, а не "*" )

  • 1
    избегайте IN, используйте EXISTS - подсказка есть в названии вопроса
  • 20
    Левое внешнее соединение, вероятно, является самым быстрым в общем случае, поскольку оно предотвращает повторное выполнение подзапроса.
Показать ещё 5 комментариев
56
SELECT Call.ID, Call.date, Call.phone_number 
FROM Call 
LEFT OUTER JOIN Phone_Book 
  ON (Call.phone_number=Phone_book.phone_number) 
  WHERE Phone_book.phone_number IS NULL

Следует удалить подзапрос, позволяя оптимизатору запросов работать с его магией.

Кроме того, избегайте "SELECT *", потому что он может нарушить ваш код, если кто-то изменяет основные таблицы или представления (и это неэффективно).

  • 9
    Как правило, это наиболее эффективный метод, поскольку он не выполняет несколько проходов на втором столе ... надеюсь, что некоторые люди читают кометы.
  • 3
    Я предпочел бы надеяться, что люди профилируют: если вы не являетесь гуру высокой производительности SQL, заранее сказать, что будет самым быстрым, довольно сложно (и зависит от используемого вами механизма СУБД).
Показать ещё 2 комментария
17

Нижеприведенный код будет немного более эффективным, чем приведенные выше ответы при работе с более крупными наборами данных.

SELECT * FROM Call WHERE 
NOT EXISTS (SELECT 'x' FROM Phone_book where 
Phone_book.phone_number = Call.phone_number)
  • 1
    Как всегда, стоит профилировать производительность запросов по отношению к целевому набору данных, чтобы выбрать наиболее эффективный. В наши дни оптимизаторы SQL достаточно хороши, поэтому результаты производительности часто удивляют.
  • 0
    Преимущество этого подхода (по сравнению с LEFT OUTER JOIN от WOPR) состоит в том, что он избегает возврата нескольких строк на строку Call , если в Phone_book есть несколько совпадающих строк. То есть, если между вашими двумя таблицами есть отношение 1:N
Показать ещё 1 комментарий
4

Я думаю,

SELECT CALL.* FROM CALL LEFT JOIN Phone_book ON 
CALL.id = Phone_book.id WHERE Phone_book.name IS NULL
  • 0
    Столбец id в таблице call не совпадает со значением столбца id в таблице Phone_book , поэтому вы не можете присоединиться к этим значениям. Смотрите ответ WOPR для аналогичного подхода.
3
SELECT DISTINCT Call.id 
FROM Call 
LEFT OUTER JOIN Phone_book USING (id) 
WHERE Phone_book.id IS NULL

Это вернет лишние идентификаторы, отсутствующие в таблице Phone_book.

1
SELECT name, phone_number FROM Call a
WHERE a.phone_number NOT IN (SELECT b.phone_number FROM Phone_book b)
  • 0
    Это не дает ответа на вопрос. Чтобы критиковать или запросить разъяснения у автора, оставьте комментарий под своим постом. - Из обзора
  • 0
    @DennisKriechel обновил запрос, чтобы он был более специфичным для вопроса.
1
SELECT t1.ColumnID,
CASE 
    WHEN NOT EXISTS( SELECT t2.FieldText  
                     FROM Table t2 
                     WHERE t2.ColumnID = t1.ColumnID) 
    THEN t1.FieldText
    ELSE t2.FieldText
END FieldText       
FROM Table1 t1, Table2 t2
  • 0
    Это вернет вам данные из одной таблицы, если данные не присутствуют в другой таблице для того же столбца
  • 0
    Не могли бы вы также добавить некоторые объяснения того, что это решает?
0

В качестве альтернативы,

select id from call
minus
select id from phone_number
  • 1
    Не уверен, что это отвечает на вопрос как есть (хотя оператор МИНУС) является новым дополнением. Это закончилось в очереди низкого качества - вы могли бы улучшить этот ответ.

Ещё вопросы

Сообщество Overcoder
Наверх
Меню