MySQL驱动表

文章目录

  • 驱动表定义
  • mysql关联查询的概念
  • 总结

驱动表定义

当进行多表连接查询时, [驱动表] 的定义为:

1)指定了(where查询)条件时,满足查询条件的记录行数少的表为[驱动表] 如图一

2)未指定(where)条件时,行数少的表为[驱动表 mysql不使用join连接,让优化器自动选择](重要); 如图二

但针对join驱动表选择如下:

先了解在join连接时哪个表是驱动表,哪个表是被驱动表:
1.当使用left join时,左表是驱动表,右表是被驱动表
2.当使用right join时,右表时驱动表,左表是驱动表
3.当使用join时,mysql会选择数据量比较小的表作为驱动表,大表作为被驱动表
图一

SELECT
	pci.id AS company_id,pci.company_name,t.station_id,pas.station_name,t.monitor_time,t.factor_id,t.monitor_status 
FROM
	ps_hour_air_data t
	LEFT JOIN ps_air_station pas ON t.station_id = pas.id
	LEFT JOIN ps_air_outlet pao ON pas.outlet_id = pao.id
	LEFT JOIN ps_company_info pci ON pao.company_id = pci.id
	LEFT JOIN ps_factor t3 ON t.factor_id = t3.id 
	AND t3.factor_type = '2' 
WHERE
	pas.STATUS = '1' 
	AND pao.STATUS = '1' 
	AND FIND_IN_SET( t.factor_id, 'fid0,fid7,fid1,fid2,fid3,fid5' ) 
	AND t.monitor_time >= '2021070100' 
	AND t.monitor_time < '2021073123' 
ORDER BY
	t.monitor_time DESC

在这里插入图片描述

图二

SELECT
	pci.id AS company_id,pci.company_name,t.station_id,pas.station_name,t.monitor_time,t.factor_id,t.monitor_status 
FROM
	ps_hour_air_data t
	LEFT JOIN ps_air_station pas ON t.station_id = pas.id
	LEFT JOIN ps_air_outlet pao ON pas.outlet_id = pao.id
	LEFT JOIN ps_company_info pci ON pao.company_id = pci.id
	LEFT JOIN ps_factor t3 ON t.factor_id = t3.id 
	AND t3.factor_type = '2' 
ORDER BY
	t.monitor_time DESC

在这里插入图片描述
注意点:当使用left join等关联查询时,若order by子句和group by子句都来自于从表时会产生临时表,来自于非驱动表的排序都是文件排序;可以试着通过改变where条件来选择表少的行作为驱动表,或者force index 来指定索引

mysql关联查询的概念

MySQL 表关联的算法是 Nest Loop Join(嵌套循环),是通过驱动表的结果集作为循环基础数据,然后一条一条地通过该结果集中的数据作为过滤条件到下一个表中查询数据,然后合并结果。

例: user表10000条数据,class表20条数据

SELECT * FROM user u LEFT JOIN class c u.userid=c.userid

这样则需要用user表循环10000次才能查询出来,而如果用class表驱动user表则只需要循环20次就能查询出来

SELECT * FROM class c LEFT JOIN user u c.userid=u.userid
小结果集驱动大结果集

de.cel 在2012年总结说,不管是你,还是 MySQL,优化的目标是尽可能减少JOIN中Nested Loop的循环次数。

以此保证:永远用小结果集驱动大结果集(Important)!

总结

  • left join 不变,对驱动表可以直接排序,对非驱动表(的字段排序)需要对循环查询的合并结果(临时表)进行排序的(Important!)
    下表的驱动表是pas,非驱动表是t,
SELECT
	pci.id AS company_id,pci.company_name,t.station_id,pas.station_name,t.monitor_time,t.factor_id,t.monitor_status 
FROM
	ps_hour_air_data t
	LEFT JOIN ps_air_station pas ON t.station_id = pas.id
	LEFT JOIN ps_air_outlet pao ON pas.outlet_id = pao.id
	LEFT JOIN ps_company_info pci ON pao.company_id = pci.id
	LEFT JOIN ps_factor t3 ON t.factor_id = t3.id 
	AND t3.factor_type = '2' 
WHERE
	pas.STATUS = '1' 
	AND pao.STATUS = '1' 
	AND FIND_IN_SET( t.factor_id, 'fid0,fid7,fid1,fid2,fid3,fid5' ) 
	AND t.monitor_time >= '2021070100' 
	AND t.monitor_time < '2021073123' 
ORDER BY
	t.monitor_time DESC

在这里插入图片描述

你可能感兴趣的:(MySQL,mysql)