本文转载自Mysql的join算法
导语
在Mysql中,使用Nested-Loop Join
的算法思想去优化join
,Nested-Loop Join
翻译成中文则是“嵌套循环连接”。
举个例子:
select * from t1 inner join t2 on t1.id=t2.tid
t1
称为外层表,也可称为驱动表。t2
称为内层表,也可称为被驱动表。
//伪代码表示:
List<Row> result = new ArrayList<>();
for(Row r1 in List<Row> t1){
for(Row r2 in List<Row> t2){
if(r1.id = r2.tid){
result.add(r1.join(r2));
}
}
}
在Mysql的实现中,Nested-Loop Join
有3种实现的算法:
Simple Nested-Loop Join
:SNLJ
,简单嵌套循环连接Index Nested-Loop Join
:INLJ
,索引嵌套循环连接Block Nested-Loop Join
:BNLJ
,缓存块嵌套循环连接
在选择Join
算法时,会有优先级,理论上会优先判断能否使用INLJ
、BNLJ
:
Index Nested-LoopJoin
>Block Nested-Loop Join
>Simple Nested-Loop Join
Simple Nested-Loop
简单嵌套循环连接实际上就是简单粗暴的嵌套循环,如果table1
有1万条数据,table2
有1万条数据,那么数据比较的次数=1万 * 1万 =1亿次,这种查询效率会非常慢。
所以Mysql继续优化,然后衍生出Index Nested-LoopJoin
、Block Nested-Loop Join
两种NLJ算法。在执行join
查询时mysql会根据情况选择两种之一进行join查询。
Index Nested-LoopJoin(减少内层表数据的匹配次数)
索引嵌套循环连接是基于索引进行连接的算法,索引是基于内层表的,通过外层表匹配条件直接与内层表索引进行匹配,避免和内层表的每条记录进行比较, 从而利用索引的查询减少了对内层表的匹配次数,优势极大的提升了 join
的性能:
原来的匹配次数 = 外层表行数 * 内层表行数
优化后的匹配次数= 外层表的行数 * 内层表索引的高度
使用场景:只有内层表join的列有索引时,才能用到Index Nested-LoopJoin
进行连接。
由于用到索引,如果索引是辅助索引而且返回的数据还包括内层表的其他数据,则会回内层表查询数据,多了一些IO操作。
Block Nested-Loop Join(减少内层表数据的循环次数)
缓存块嵌套循环连接通过一次性缓存多条数据,把参与查询的列缓存到Join Buffer 里,然后拿join buffer里的数据批量与内层表的数据进行匹配,从而减少了内层循环的次数(遍历一次内层表就可以批量匹配一次Join Buffer里面的外层表数据)。
当不使用Index Nested-Loop Join
的时候,默认使用Block Nested-Loop Join
。
什么是Join Buffer
?
Join Buffer
会缓存所有参与查询的列而不是只有Join的列。- 可以通过调整
join_buffer_size
缓存大小 join_buffer_size
的默认值是256K,join_buffer_size
的最大值在MySQL 5.1.22
版本前是4G
,而之后的版本才能在64位操作系统下申请大于4G
的Join Buffer
空间。- 使用
Block Nested-Loop Join
算法需要开启优化器管理配置的optimizer_switch
的设置block_nested_loop
为on
,默认为开启。
如何优化Join速度
- 用小结果集驱动大结果集,减少外层循环的数据量:
- 如果小结果集和大结果集连接的列都是索引列,
mysql
在内连接时也会选择用小结果集驱动大结果集,因为索引查询的成本是比较固定的,这时候外层的循环越少,join
的速度便越快。 - 为匹配的条件增加索引:争取使用
INLJ
,减少内层表的循环次数 - 增大
join buffer size
的大小:当使用BNLJ
时,一次缓存的数据越多,那么外层表循环的次数就越少 - 减少不必要的字段查询:
- 当用到
BNLJ
时,字段越少,join buffer
所缓存的数据就越多,外层表的循环次数就越少; - 当用到
INLJ
时,如果可以不回表查询,即利用到覆盖索引,则可能可以提示速度。(未经验证,只是一个推论)
- 当用到
参考文档
https://www.wengbi.com/thread_99558_1.html
https://www.cnblogs.com/starhu/p/6418842.html
https://www.cnblogs.com/starhu/p/6418833.html