SQL ON与WHERE区别

在 MySQL 中,ONWHERE 都用于过滤数据,但它们的使用场景和作用有所不同,尤其是在涉及 JOIN 操作时。下面通过具体的例子来说明它们的区别。

1. ON 的作用

ON 用于指定表之间的连接条件,决定哪些行应该被连接在一起。它在 JOIN 操作时生效。

示例:

假设有两个表:

orders 表:存储订单信息。

复制代码
+---------+------------+-------------+
| order_id| order_date | customer_id |
+---------+------------+-------------+
| 1       | 2023-01-01 | 101         |
| 2       | 2023-02-01 | 102         |
| 3       | 2023-03-01 | 103         |
+---------+------------+-------------+

customers 表:存储客户信息。

复制代码
+-------------+------------+
| customer_id | name       |
+-------------+------------+
| 101         | Alice      |
| 102         | Bob        |
| 103         | Charlie    |
+-------------+------------+

使用 ON 进行连接:

复制代码
SELECT *
FROM orders
JOIN customers ON orders.customer_id = customers.customer_id;

结果

复制代码
+---------+------------+-------------+-------------+-------+
| order_id| order_date | customer_id | customer_id | name  |
+---------+------------+-------------+-------------+-------+
| 1       | 2023-01-01 | 101         | 101         | Alice |
| 2       | 2023-02-01 | 102         | 102         | Bob   |
| 3       | 2023-03-01 | 103         | 103         | Charlie|
+---------+------------+-------------+-------------+-------+

ON 条件指定了 orders.customer_id = customers.customer_id,决定了哪些行应该被连接在一起。

2. WHERE 的作用

WHERE 用于过滤查询结果集中的行。它在连接操作之后生效。

示例:

继续使用上面的 orderscustomers 表,现在需要查询 order_date 大于 2023-01-01 的订单。

复制代码
SELECT *
FROM orders
JOIN customers ON orders.customer_id = customers.customer_id
WHERE orders.order_date > '2023-01-01';

结果

复制代码
+---------+------------+-------------+-------------+-------+
| order_id| order_date | customer_id | customer_id | name  |
+---------+------------+-------------+-------------+-------+
| 2       | 2023-02-01 | 102         | 102         | Bob   |
| 3       | 2023-03-01 | 103         | 103         | Charlie|
+---------+------------+-------------+-------------+-------+
  • WHERE 条件过滤了 order_date 大于 2023-01-01 的记录。

3. ONWHERE 的区别

场景 1:INNER JOIN 中的 ONWHERE

INNER JOIN 中,ONWHERE 的效果通常是相同的,因为 INNER JOIN 只返回满足连接条件的行。

示例

复制代码
-- 使用 ON 条件
SELECT *
FROM orders
JOIN customers ON orders.customer_id = customers.customer_id
AND orders.order_date > '2023-01-01';

-- 使用 WHERE 条件
SELECT *
FROM orders
JOIN customers ON orders.customer_id = customers.customer_id
WHERE orders.order_date > '2023-01-01';

结果

复制代码
+---------+------------+-------------+-------------+-------+
| order_id| order_date | customer_id | customer_id | name  |
+---------+------------+-------------+-------------+-------+
| 2       | 2023-02-01 | 102         | 102         | Bob   |
| 3       | 2023-03-01 | 103         | 103         | Charlie|
+---------+------------+-------------+-------------+-------+

INNER JOIN 中,ONWHERE 的结果是相同的。

场景 2:LEFT JOIN 中的 ONWHERE

LEFT JOIN 中,ONWHERE 的效果可能不同,因为 LEFT JOIN 会保留左表中的所有行,即使右表中没有匹配的行。

示例

假设 orders 表中有一条记录没有对应的 customer_id

复制代码
+---------+------------+-------------+
| order_id| order_date | customer_id |
+---------+------------+-------------+
| 1       | 2023-01-01 | 101         |
| 2       | 2023-02-01 | 102         |
| 3       | 2023-03-01 | 103         |
| 4       | 2023-04-01 | NULL        | -- 没有对应的 customer_id
+---------+------------+-------------+

使用 ON 条件

复制代码
SELECT *
FROM orders
LEFT JOIN customers ON orders.customer_id = customers.customer_id
AND customers.name = 'Alice';

结果

复制代码
+---------+------------+-------------+-------------+-------+
| order_id| order_date | customer_id | customer_id | name  |
+---------+------------+-------------+-------------+-------+
| 1       | 2023-01-01 | 101         | 101         | Alice |
| 2       | 2023-02-01 | 102         | NULL        | NULL  |
| 3       | 2023-03-01 | 103         | NULL        | NULL  |
| 4       | 2023-04-01 | NULL        | NULL        | NULL  |
+---------+------------+-------------+-------------+-------+

ON 条件保留了所有 orders 表中的行,即使 customers.name 不是 Alice

使用 WHERE 条件

复制代码
SELECT *
FROM orders
LEFT JOIN customers ON orders.customer_id = customers.customer_id
WHERE customers.name = 'Alice';

结果

复制代码
+---------+------------+-------------+-------------+-------+
| order_id| order_date | customer_id | customer_id | name  |
+---------+------------+-------------+-------------+-------+
| 1       | 2023-01-01 | 101         | 101         | Alice |
+---------+------------+-------------+-------------+-------+

WHERE 条件过滤了结果集,只返回 customers.name = 'Alice' 的行

4. 总结

  • ON:用于指定连接条件,决定哪些行应该被连接在一起。它在连接操作时生效。

  • WHERE:用于过滤查询结果,决定哪些行应该被返回。它在连接操作之后生效。

  • INNER JOIN 中,ONWHERE 的效果通常是相同的。

  • LEFT JOINRIGHT JOIN 中,ONWHERE 的效果可能不同,需要根据实际需求谨慎选择。

相关推荐
lzhdim1 小时前
SQL 入门 15:SQL 事务:从 ACID 到四种常见的并发问题
数据库·sql
瀚高PG实验室2 小时前
瀚高企业版V9.1.1在pg_restore还原备份文件时提示extract函数语法问题
数据库·瀚高数据库
TDengine (老段)2 小时前
TDengine Tag 设计哲学与 Schema 变更机制
大数据·数据库·物联网·时序数据库·iot·tdengine·涛思数据
YOU OU3 小时前
Spring IoC&DI
java·数据库·spring
Muscleheng4 小时前
Navicat连接postgresql时出现‘datlastsysoid does not exist‘报错
数据库·postgresql
罗超驿5 小时前
18.事务的隔离性和隔离级别:MySQL面试高频考点全解析
数据库·mysql·面试
jran-5 小时前
Redis 命令
数据库·redis·缓存
小江的记录本5 小时前
【Java基础】Java 8-21新特性:JDK21 LTS:虚拟线程、模式匹配switch、结构化并发、序列集合(附《思维导图》+《面试高频考点清单》)
java·数据库·python·mysql·spring·面试·maven
June`6 小时前
多线程redis下如何解决aof重写和rdb持久化的数据一致性问题
数据库·redis·缓存
二宝哥6 小时前
离线安装maven
java·数据库·maven