时间戳转换器

从一个表中查找另一个表中不存在的记录

日期:2023-02-24     浏览:85    
【中文标题】从一个表中查找另一个表中不存在的记录【英文标题】:Find records from one table which don't exist in another 【发布时间】:2010-09-26 22:35:09 【问题描述】:

我有以下两个表(在 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 |
+----+------+--------------+

【问题讨论】:

【参考方案1】:

有几种不同的方法可以做到这一点,效率各不相同,具体取决于您的查询优化器有多好,以及您的两个表的相对大小:

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

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

(忽略这一点,正如其他人所说,通常最好只选择您想要的列,而不是'*')

【讨论】:

避免 IN,使用 EXISTS - 提示在问题标题中 左外连接在一般情况下可能是最快的,因为它可以防止重复执行子查询。 不要挑剔,但我建议的子查询返回 select 'x' 而不是 select * 是的 - MySQL 手册建议这对于 'EXISTS' 查询是正常的 @Alnitak:在第二个查询中,子查询中不需要SELECT *。相反,例如,SELECT 1 应该足够漂亮。【参考方案2】:
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 *”,因为如果有人更改基础表或视图,它可能会破坏您的代码(而且效率低下)。

【讨论】:

这通常是最有效的方法,因为它不会在第二张桌子上执行多次传递......希望有些人正在阅读评论。 我宁愿希望人们介绍一下:除非您是*** SQL 性能专家,否则提前告诉什么最快是相当困难的(并且取决于您使用的 DBMS 引擎)。跨度> Big O 符号很容易告诉你在这种情况下你可以期望最快的速度。数量级不同。 如果您的两个表之间存在1:N 关系,请参见Afterlife's answer 和我的评论。或添加DISTINCT,如Vlado's answer中所示【参考方案3】:

在处理更大的数据集时,下面的代码会比上面给出的答案更有效。

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

【讨论】:

与往常一样,有必要针对目标数据集分析查询的性能,以选择性能最佳的一个。如今,SQL 优化器已经足够好,以至于性能结果常常令人惊讶。 这种方法的一个优点(与 WOPR 的 LEFT OUTER JOIN 相比)是,如果 Phone_book 中有多个匹配行,它可以避免在 Call 的每行返回多行。也就是说,如果您的两个表之间存在1:N 关系。 我会从这个开始——它直接代表了意图。如果性能不够好,请确保存在适当的索引。只有这样,尝试不太明显的LEFT OUTER JOIN,看看它的性能是否更好。【参考方案4】:
SELECT DISTINCT Call.id 
FROM Call 
LEFT OUTER JOIN Phone_book USING (id) 
WHERE Phone_book.id IS NULL

这将返回您的 Phone_book 表中缺少的额外 ID。

【讨论】:

【参考方案5】:

我觉得

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

【讨论】:

call 表中的id 列与Phone_book 表中的id 列的值不同,因此您无法加入这些值。有关类似方法,请参阅 WOPR 的答案。【参考方案6】:
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

【讨论】:

如果同一列的另一个表中没有数据,这将返回一个表中的数据【参考方案7】:
SELECT name, phone_number FROM Call a
WHERE a.phone_number NOT IN (SELECT b.phone_number FROM Phone_book b)

【讨论】:

这并没有提供问题的答案。要批评或要求作者澄清,请在他们的帖子下方留下评论。 - From Review @DennisKriechel 更新了查询,使其更具体地针对问题。【参考方案8】:

或者,

select id from call
minus
select id from phone_number

【讨论】:

不确定这是否能按原样回答问题(尽管 MINUS)运算符是新增的。这最终进入了低质量队列 - 您可能希望增强此答案。【参考方案9】:

别忘了检查你的索引!

如果您的表格很大,您需要确保电话簿在phone_number 字段上有索引。对于大型表,数据库很可能会选择同时扫描这两个表。

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

您应该同时创建包含phone_number 的Phone_Book 和Call 索引。如果性能成为问题,请尝试这样的精简索引,仅使用电话号码:

字段越少越好,因为它必须完全加载它。您需要两个表的索引。

ALTER TABLE [dbo].Phone_Book ADD CONSTRAINT [IX_Unique_PhoneNumber] UNIQUE NONCLUSTERED 
(
    Phone_Number
)
WITH (STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ONLINE = ON) ON [PRIMARY]
GO

如果您查看查询计划,它将看起来像这样,您可以确认您的新索引实际上正在被使用。请注意,这适用于 SQL Server,但对于 MySQL 应该类似。

对于我展示的查询,除了扫描两个表中的每条记录之外,实际上没有其他方法可以让数据库产生结果。

【讨论】:

相关文章

选择另一个表中不存在的值

{】选择另一个表中不存在的值【英文标题】:selectavaluewhereitdoesn\'texistinanothertable【发布时间】:2011-02-2717:59:14【问题描述】:我有两张桌子表A:ID1234表B:ID123我有两个请求:我想选择表A中表B没有的所有行,在本例中为第4行。...}

Laravel - 从一个表中获取不存在于另一个表中的记录,并附加了 where 子句

{】Laravel-从一个表中获取不存在于另一个表中的记录,并附加了where子句【英文标题】:Laravel-Getrecordsfromonetablethatdoesn\'texistinanotherwithawhereclauseattached【发布时间】:2018-05-2809:46:44【问题描述】:我有以下SQL表(在MySQL中):student...}

mysql获取条件中描述的所有记录,即使表中不存在

{...ble【发布时间】:2015-02-1902:05:30【问题描述】:假设我有一个如下表:我的表+---------------+|id|name|+---------------+|1|ABC|+---------------+|2|XYZ|}

MySQL:如果表中不存在则插入记录[重复]

{】MySQL:如果表中不存在则插入记录[重复]【英文标题】:MySQL:Insertrecordifnotexistsintable[duplicate]【发布时间】:2011-03-1100:29:43【问题描述】:我正在尝试执行以下查询:INSERTINTOtable_listnames(name,address,tele)VALUES(\'Rupert\',\'Somewhere\',\'022...}

如何获取一个MySql表中存在的记录而不是另一个[重复]

{】如何获取一个MySql表中存在的记录而不是另一个[重复]【英文标题】:HowtogetrecordsthatexistinoneMySqltableandnotanother[duplicate]【发布时间】:2013-09-1405:53:09【问题描述】:我需要编写一个查询来返回存在于一个MySql表中但不存在于另一...}

获取表中不存在的 ID 列表

{...ble【发布时间】:2013-04-0317:25:54【问题描述】:假设我有一个id列表,例如(1,3,9,2,4,86)和一个带有列id的表。我想查找列表中没有匹配行的所有数字。即如果mysql表是这样的:idletter1a2b3c4d5e6f7g我有列表(1,3,9,2,4,86),我想要一个返回(...}

MySQL内部连接查询以获取其他表中不存在的记录

{】MySQL内部连接查询以获取其他表中不存在的记录【英文标题】:MySQLInnerJoinQueryToGetRecordsNotPresentinOtherTable【发布时间】:2012-04-1420:00:28【问题描述】:我有表1,all_countries,如下-id|country------------------1|USA2|China3|India4|France5|UK6|Aust...}

数据结构与算法:查找(代码片段)

{查找是在大量的信息中寻找一个特定的信息元素,在计算机应用中,查找是常用的基本运算,例如编译程序中符号表的查找。定义:根据给定的某个值,在查找表中确定一个其关键字等于给定值的数据元素(或记录)。分类:静...}

在一个表中查找不存在于另一个表中的id(代码片段)

{给出了卡桑德拉的两张桌子:xs(idbigint,xbigint)0,101,112,123,134,14ys(idbigint,ybigint)0,1002,1444,196在这个最小的例子中,我想得到xs的所有行,其中id尚未包含在ys中,即select1,113,13我怎样才能做到这一点?根据我想在ys中创建新行的选定行...}

从 SQL SELECT 语句中获取值,即使它在表中不存在

{...statement【发布时间】:2012-02-2719:37:53【问题描述】:我有一个如下所示的MySQL表:|---ID---|---COUNTER---||1|2||2|6||3|1||5|9||6|10|我正在寻找返回ID及其COUNTER的SELECT语句。该表}

Copyright ©2021 时间戳转换器 小常识 114pp | 陕ICP备18005036号