外连接的两种写法的总结

最近发现自己对外关联比较模糊,特别是使用oracle老风格的(+)与ANSI风格的外关联的时候。今天下午没什么事情自己整理了一下子。

--无其他条件
select * from n_fwy_a a left join n_fwy_b b on a.id=b.id order by a.id;
select * from n_fwy_a a ,n_fwy_b b where a.id=b.id(+) order by a.id;
这两重写法的查询的结果一致
--在主表加附加条件
select * from n_fwy_a a ,n_fwy_b b where a.id=b.id(+) and a.id >5 order by a.id;
select * from n_fwy_a a left join n_fwy_b b on a.id=b.id where a.id>5 order by a.id;
select * from n_fwy_a a left join n_fwy_b b on a.id=b.id and a.id>5 order by a.id;--no
这三种写法的前两种查询结果一致,用第2种sql查询得的结果发现出现了n_fwy_a的全集数据
--在附表中加条件
select * from n_fwy_a a ,n_fwy_b b where a.id=b.id(+) and b.id(+) >5 order by a.id;
select * from n_fwy_a a left join n_fwy_b b on a.id=b.id where b.id >5 order by a.id;--no
select * from n_fwy_a a left join n_fwy_b b on a.id=b.id and b.id >5 order by a.id;
这三种写法的前1,3种查询结果一致,用第2种sql查询得的结果发现仅仅出现了b.id >5的数据集,而其他两种结果集合是a的全集
--对于in 和 or 操作
在老风格(+)的外关联操作中,where 条件中是不允许附表引用in和or操作的。
--对于in操作
select *
from n_fwy_a a ,
(select * from n_fwy_b b where b.id in (2,8)) b
where a.id=b.id (+);
select *
from n_fwy_a a left join n_fwy_b b on a.id=b.id and b.id in (2,8);
对于在附表中引用in和or操作,两种写法所得的结果一致
select *
from (select * from n_fwy_a a where a.id in (2,9)) a ,
n_fwy_b b
where a.id=b.id (+);
select *
from n_fwy_a a left join n_fwy_b b on a.id=b.id and a.id in (2,9);
对于在主表中引用in和or操作,两种写法所得的结果就不一致了,和好理解第一种写法的SQL把主表的数据集变小了,第二种写法的sql的数据集还是主表的全集
--对于OR操作与in的一致

总体结论:对于ANSI风格的外关联其on后条件是关联前的筛选数据条件,但是无论加什么条件,其结果集大小与主表的数据集一样大,对于其where后的条件是关联之后的条件。
     对于(+)风格的外关联,where后的条件是关联后的筛选数据的条件。针对附表不能使用in和or操作,若要使用应该在关联之前使用,以缩小数据集合(即在from 中使用表查询)。
     否则,起不到外关联的作用,结果集为内连接的结果

[@more@]

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/519130/viewspace-964987/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/519130/viewspace-964987/

<think>好的,用户现在问的是SQL内连接的写法。首先,我需要回忆一下内连接的基本概念。内连接(INNER JOIN)是SQL中用于从两个或多个表中获取满足连接条件的行的操作,只返回两个表中匹配的行。 首先,应该解释内连接的基本语法。标准的写法是使用JOIN关键字,后面跟ON子句指定连接条件。另外,还有一种隐式的写法,使用逗号分隔表,然后在WHERE子句中指定条件,不过现在更推荐显式的JOIN语法,因为更清晰。 然后,需要给出具体的示例。比如,假设有两个表,employees和departments,employees有department_id,departments有id和name。内连接这两个表,获取员工及其部门名称。 还要注意别名的使用,这在多表连接时很常见,可以简化代码。比如,employees别名为e,departments别名为d。 另外,需要说明内连接的特点:只返回两个表中连接条件匹配的行。如果某行在一个表中存在,但另一个表中没有匹配的行,那么这行不会被包含在结果中。 可能还需要提到不同的数据库是否有差异,但通常标准SQL的内连接写法是通用的。 然后,用户可能需要了解内连接与其他连接(如左连接、右连接)的区别,但用户的问题主要是写法,所以重点放在语法上。 最后,总结内连接的关键要素:JOIN关键字、ON子句的连接条件,以及结果的特点。 现在需要组织这些内容,确保清晰易懂,示例正确,并覆盖基本用法和常见情况。 </think>在 SQL 中,**内连接(INNER JOIN)** 用于从两个或多个表中获取满足连接条件的行,仅返回**同时在两个表中匹配**的记录。它是最常用的连接类型,适用于需要关联多表数据的场景(如用户表与订单表关联查询用户及其订单)。以下是内连接的标准写法、核心语法及示例说明: ### **一、内连接的标准语法** 内连接的显式写法使用 `JOIN` 关键字(可简写为 `INNER JOIN`,两者等价),并通过 `ON` 子句定义连接条件。语法格式如下: ```sql SELECT 列名1, 列名2, ... FROM 表A JOIN 表B ON 表A.关联列 = 表B.关联列; ``` ### **二、关键要素说明** 1. **表关联条件**:`ON` 子句必须指定两个表的关联逻辑(如 `用户表.id = 订单表.user_id`),确保数据正确关联。 2. **列名冲突处理**:若两表存在同名列(如都有 `id`),需用表名/别名明确指定(如 `表A.id` 或 `别名.id`)。 3. **别名优化**:为表起别名(如 `用户表 AS u`)可简化代码,尤其适用于多表连接场景。 ### **三、示例演示** 假设有以下两张表: #### **表1:users(用户表)** | id | name | age | |----|--------|-----| | 1 | Alice | 25 | | 2 | Bob | 30 | | 3 | Charlie| 28 | #### **表2:orders(订单表)** | order_id | user_id | amount | |----------|---------|--------| | 101 | 1 | 150 | | 102 | 2 | 200 | | 103 | 4 | 300 | #### **需求**:查询所有有订单的用户及其订单金额(仅匹配 `users.id = orders.user_id` 的记录)。 #### **写法1:显式 JOIN(推荐)** ```sql SELECT u.name, -- 使用别名避免列名冲突 o.amount FROM users AS u -- 为 users 表起别名 u JOIN orders AS o -- 为 orders 表起别名 o ON u.id = o.user_id; -- 关联条件:用户ID匹配订单用户ID ``` #### **写法2:隐式 JOIN(旧语法,不推荐)** 通过 `WHERE` 子句隐式实现内连接(等价于显式 `JOIN`,但可读性差): ```sql SELECT users.name, orders.amount FROM users, orders -- 逗号分隔表(隐式连接) WHERE users.id = orders.user_id; -- 连接条件写在 WHERE 中 ``` #### **执行结果** | name | amount | |--------|--------| | Alice | 150 | | Bob | 200 | ### **四、注意事项** 1. **仅返回匹配行**:若 `users` 表中某用户无订单(如 `Charlie`,id=3),或 `orders` 表中某订单无对应用户(如 `order_id=103`,user_id=4),则这些记录不会出现在结果中。 2. **多表连接**:内连接支持多表级联(如 `表A JOIN 表B ON ... JOIN 表C ON ...`),需逐层定义关联条件。 3. **过滤条件**:可通过 `WHERE` 子句进一步过滤结果(如 `WHERE o.amount > 180`)。 ### **五、内连接 vs 其他连接** | 连接类型 | 特点 | |----------------|---------------------------------------| | **INNER JOIN** | 仅返回两表**完全匹配**的行(最严格)。| | LEFT JOIN | 返回左表所有行,右表无匹配时用 `NULL` 填充。 | | RIGHT JOIN | 返回右表所有行,左表无匹配时用 `NULL` 填充。 | ### **总结** 内连接的核心是通过 `JOIN ... ON` 显式关联多表,仅保留匹配的行。实际开发中推荐使用显式 `JOIN` 语法,以提高代码可读性和维护性。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符  | 博主筛选后可见
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值