SQL - 找到一个表中的记录不存在,在另一

SQL - 找到一个表中的记录不存在,在另一

问题描述:

我已经得到了以下两个SQL表(在MySQL):SQL - 找到一个表中的记录不存在,在另一

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 | 
+----+------+--------------+ 

任何帮助将不胜感激。

有几种不同的方式这样做的,有不同的效率,这取决于您的查询优化器有多好,你的两个表的相对大小:

这是最短的语句,如果可能是最快的电话本是很短:

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

避免,使用EXISTS - 该提示是在问题的标题 – annakata 2008-12-15 12:02:11

+16

左外连接可能是最快的,一般情况下,它可以防止重复的执行子查询。 – WOPR 2008-12-15 21:40:40

+0

不挑剔,但我的建议子查询返回select 'x'而不是select * Alterlife 2008-12-17 08:36:39

下面的代码会比上面给出的答案多一点效率当处理更大的数据集时。

SELECT * FROM Call WHERE 
NOT EXISTS (SELECT 'x' FROM Phone_book where 
Phone_book.phone_number = Call.phone_number) 
+1

与往常一样,根据目标数据集查询性能可以选择具有最佳性能的查询。 SQL优化器现在已经足够好了,以至于性能结果往往令人惊讶。 – 2008-12-15 09:44:33

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 *”,因为如果有人更改了底层表或视图(并且效率低下),它可能会破坏您的代码。

+6

这通常是最有效的方法,因为它不会在第二张桌子上执行多次传递......希望有些人阅读comemnts。 – Nerdfest 2008-12-15 11:57:27

我觉得

SELECT CALL.* FROM CALL LEFT JOIN Phone_book ON 
CALL.id = Phone_book.id WHERE Phone_book.name IS NULL 

SELECT DISTINCT Call.id 
FROM Call 
LEFT OUTER JOIN Phone_book USING (id) 
WHERE Phone_book.id IS NULL 

这将返回缺少额外的ID,在你的Phone_book表。

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 

SELECT name, phone_number FROM Call a 
WHERE a.phone_number NOT IN (SELECT b.phone_number FROM Phone_book b) 

另外,

select id from call 
minus 
select id from phone_number