sql多表关联查询使用JOIN..ON与where的优化场景

先说明原因:

优化的本质就是(join on 和where的执行顺序)

关键字: on

数据库在通过连接两张或多张表来返回记录时,都会生成一张中间的临时表,然后再将这张临时表返回给用户。

在使用left jion时,on和where条件的区别如下:

1、 on条件是在生成临时表时使用的条件,它不管on中的条件是否为真,都会返回左边表中的记录。

2、where条件是在临时表生成好后,再对临时表进行过滤的条件。这时已经没有left join的含义(必须返回左边表的记录)了,条件不为真的就全部过滤掉。

假设有两张表:

表1:tab2

idsize
110
220
330

表2:tab2

sizename
10AAA
20BBB
20CCC

两条SQL:
1、select * form tab1 left join tab2 on (tab1.size = tab2.size) where tab2.name=’AAA’
2、select * form tab1 left join tab2 on (tab1.size = tab2.size and tab2.name=’AAA’)

第一条SQL的过程:

 

1、中间表
on条件:
tab1.size = tab2.size
tab1.idtab1.sizetab2.sizetab2.name
11010AAA
22020BBB
22020CCC
330(null)(null)

 

  
2、再对中间表过滤
where 条件:
tab2.name=’AAA’
tab1.idtab1.sizetab2.sizetab2.name
11010AAA

 

  

 

第二条SQL的过程:

 

1、中间表
on条件:
tab1.size = tab2.size and tab2.name=’AAA’
(条件不为真也会返回左表中的记录)
tab1.idtab1.sizetab2.sizetab2.name
11010AAA
220(null)(null)
330(null)(null)

 

 


实战练习

1.首先先建立两个表

create table table1
(
  mon varchar2(14),
  dep number(4),
  yj number(4)
)
insert into table1 values ('一月份',1,10);
insert into table1 values ('一月份',2,10);
insert into table1 values ('一月份',3,5);
insert into table1 values ('二月份',2,8);
insert into table1 values ('二月份',4,9);
insert into table1 values ('二月份',3,8);

create table table2
(
  dep number(4),
  dname varchar2(30)
)

insert into table2 values(1,'国内业务一部');
insert into table2 values(2,'国内业务二部');
insert into table2 values(3,'国内业务三部');
insert into table2 values(4,'国际业务部');

2.要求查出每个部门一月份,二月份的业绩

查出结果如下


3 用以下where语句增加最后的过滤条件在特定场景下肯定是可以的 (但是鉴于这里是要1~4月全部展示,所以where子句还是不符合条件,在这里列出是为了说明where是在on条件执行之后增加过滤条件)

select t0.dep as dep,t0.dname as 部门名称 , t1.yj as 一月份业绩 , t2.yj as 二月份业绩
from table2 t0
  left join  table1 t1 on  t0.dep=t1.dep
  left join table1 t2 on  t0.dep=t2.dep
where t1.mon='一月份' and t2.mon='二月份'
order by t0.dep

4. 接下来把where条件去掉看查出来的数据有10条,可以发现这其中没有 t1.mon='一月份' and t2.mon='二月份'的限定条件,

    table2表的国际业务一部 (dep=1)与 table1表的一月份(dep=1)的业绩为10(yj=10) 匹配2次,分别匹配出一条一月份的业绩和二月份的业绩 ,因为国际业务一部二月份么有业绩,所以这明显是错误的匹配 (增加了额外的中间表数量)

5. 如果能在匹配之前先对应加上t1.mon='一月份' 或者  t2.mon='二月份'的限定条件,如果没有满足限定条件,后面的t0.dep=t1.dep就不会执行,这样就会至少在中间表中会少一条数据, 假如再使用where过滤就会减少过滤操作数量

select t0.dep as dep,t0.dname as 部门名称 , t1.yj as 一月份业绩 , t2.yj as 二月份业绩
from table2 t0
  left join  table1 t1 on t1.mon='一月份' and t0.dep=t1.dep
  left join table1 t2 on t2.mon='二月份' and t0.dep=t2.dep
order by t0.dep

  


参考文章:https://www.cnblogs.com/wlzhang/p/4532587.html

另附 提高SQL查询效率的30种方法 传送门

https://www.cnblogs.com/lykbk/p/aefweere45454545454.html

<think>好的,我需要回答用户关于SQL Server中JOIN ON语法的问题,包括作用、使用场景和示例。首先,我得回顾一下JOIN的基本概念,然后整理不同类型JOIN的区别,比如INNER JOIN、LEFT JOIN等。用户提供的引用中有提到LEFT JOIN的语法示例,可以引用那个例子。同时,引用4提到了ON条件后接AND和WHERE的区别,这部分也需要解释清楚。 首先,JOIN ON的主要作用是根据两个的共同列连接数据。需要说明它用于关联,形成临时结果集。使用场景包括需要从组合数据的查询,比如订单和客户信息关联。然后,要给出每种JOIN类型的示例,特别是用户提到的LEFT JOIN,并且用代码块展示。 接下来,要注意用户提供的引用内容,比如引用2中的LEFT JOIN示例,需要正确引用并标注[^2]。同时,关于ONWHERE的区别,引用4提到过,这部分需要详细说明,比如ON是连接条件,WHERE是过滤结果集。 还要检查是否需要遵守系统级指令,比如数学达式用$...$,但这个问题可能不涉及数学公式,所以可能不需要。但需要确保中文回答,结构清晰,分点说明。最后生成相关问题,可能涉及其他JOIN类型、性能优化、实际应用案例等。 现在需要组织内容,确保每个部分都涵盖,并正确引用提供的资料。可能的结构是: 1. 作用:关联,组合数据 2. 使用场景查询,统计分析,数据整合 3. 语法示例:不同JOIN类型的例子,特别是LEFT JOIN 4. ONWHERE的区别:引用4的解释 5. 其他注意事项:索引优化,性能影响(引用3提到ALTER INDEX,可能相关) 最后生成相关问题,比如不同JOIN的区别,优化方法,实际案例等。</think>### SQL Server中JOIN ON语法详解 #### 一、作用 `JOIN ON` 是 SQL Server 中用于**关联**的关键语法,通过指定连接条件将不同中的行组合成临时结果集。其核心作用包括: - **数据关联**:基于共同列(如外键)将的数据横向拼接。 - **结果筛选**:通过 `ON` 后的条件过滤有效关联数据[^4]。 #### 二、使用场景 1. **数据查询**:例如查询订单关联客户以获取客户姓名。 2. **统计分析**:如统计每个部门的销售额时需要连接部门和销售。 3. **数据整合**:将分散在不同中的信息合并为完整视图(如商品信息+库存信息)。 #### 三、语法示例 ##### 1. 基础语法 ```sql SELECT 列名 FROM 1 [INNER/LEFT/RIGHT/FULL] JOIN 2 ON 1.共同列 = 2.共同列; ``` ##### 2. 不同JOIN类型示例 - **INNER JOIN**(默认):仅返回匹配的行。 ```sql SELECT Orders.OrderID, Customers.CustomerName FROM Orders INNER JOIN Customers ON Orders.CustomerID = Customers.CustomerID; ``` - **LEFT JOIN**:返回左所有行,右无匹配时填充`NULL`。 ```sql SELECT * FROM 1 LEFT JOIN 2 ON 1.CommonColumn = 2.CommonColumn; ``` - **RIGHT JOIN**:LEFT JOIN相反,返回右所有行。 - **FULL JOIN**:返回左右全部行,无匹配时填充`NULL`。 #### 四、`ON``WHERE`的区别 - **ON**:定义连接时的匹配条件,影响关联过程。 - **WHERE**:在连接完成后过滤结果集。 例如,以下查询会保留左所有行,但仅筛选右满足条件的匹配: ```sql SELECT * FROM A LEFT JOIN B ON A.ID = B.ID AND B.Status = 'Active' -- ON中定义连接和部分过滤 WHERE A.Amount > 100; -- WHERE过滤最终结果 ``` #### 五、性能注意事项 1. **索引优化**:确保连接列(如`CustomerID`)已建立索引,可加速关联过程[^3]。 2. **避免过JOIN**:超过3个的连接可能影响性能,需评估是否需要冗余字段或临时
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值