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

У меня есть следующие две таблицы 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 ]

375 голосов
/ 15 декабря 2008

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

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

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

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

75 голосов
/ 15 декабря 2008
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 *», потому что он может сломать ваш код, если кто-то изменит базовые таблицы или представления (и это неэффективно).

23 голосов
/ 15 декабря 2008

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

SELECT * FROM Call WHERE 
NOT EXISTS (SELECT 'x' FROM Phone_book where 
Phone_book.phone_number = Call.phone_number)
5 голосов
/ 16 января 2013
SELECT DISTINCT Call.id 
FROM Call 
LEFT OUTER JOIN Phone_book USING (id) 
WHERE Phone_book.id IS NULL

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

3 голосов
/ 15 февраля 2012

Я думаю

SELECT CALL.* FROM CALL LEFT JOIN Phone_book ON 
CALL.id = Phone_book.id WHERE Phone_book.name IS NULL
2 голосов
/ 24 октября 2013
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
1 голос
/ 21 августа 2017

С другой стороны,

select id from call
minus
select id from phone_number
1 голос
/ 09 декабря 2015
SELECT name, phone_number FROM Call a
WHERE a.phone_number NOT IN (SELECT b.phone_number FROM Phone_book b)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...