解析MySQL join查询的原理

 更新时间:2022年01月27日 15:57:22   作者:CaptainCats  
这篇文章主要介绍了MySQL join查询的原理,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

MySQL用Nested-Loop Join算法实现join查询

区分驱动表和被驱动表,以驱动表的结果集为循环的基础,访问被驱动表过滤数据,然后合并结果,驱动表在外循环、被驱动表在内循环。
如果还有第三张参与join查询的表,则以合并的结果为驱动表,第三张表作为被驱动表,以此类推。

left join中的左表是驱动表、右表是被驱动表,right join刚好相反。

Nested-Loop Join有三种实现

SNLJ

Simple Nested-Loop Join

假设A是驱动表,B是被驱动表。

这里会扫描A表,用A的结果集作为外循环,
每循环一次,会扫描B表一遍(遍历内循环)

A表有N行,B表有M行。

SNLJ的开销如下(最大情况下):

扫描A表1次;
扫描B表N次。
总共读取记录数:N + N * M。

为了专注于理解Nested-Loop Join,这里不讨论带where子句的情况,以下相同。

BNLJ

Block Nested-Loop Join

假设A是驱动表,B是被驱动表。

用来join的字段在被驱动表没有建立索引

Join Buffer
MySQL会将驱动表结果集中(多条记录)用来join的字段缓存到Join Buffer,
Join Buffer的特点是只需要扫描被驱动表一次,就能得到Join Buffer中所有记录的匹配结果,
减少扫描的次数。

Join Buffer默认大小256k,会生成n-1个Join Buffer缓冲区,n为参与join查询的表数量。

A表有N行,B表有M行。

BNLJ的开销如下(最大情况下):

扫描A表1次;
扫描B表X次;
X的大小取决于N、join字段的大小、Join Buffer的大小,通常X<<N。

INLJ

Index Nested-Loop Join

假设A是驱动表,B是被驱动表。

用来join的字段在被驱动表建立了索引

聚集索引

非聚集索引

在这里我们假设您已对MySQL的索引结构有了一定的了解,
如果没有的话,可以去看下:通过B+Tree平衡多叉树理解InnoDB引擎的聚集和非聚集索引

这里会扫描A表,用A的结果集作为外循环,
然后通过B表的索引来检索,不会遍历B表。

A表有N行,B表有M行。

INLJ的开销如下(最大情况下):

扫描A表1次;
通过B表索引检索N次,成本比扫描B表N次会低很多;
回表:先找到非聚集索引,再找到聚集索引,会多一次磁盘IO。

NLJ优先级

INLJ>BNLJ>SNLJ

如何优化join查询效率

尽量将小表作为驱动表,大表作为被驱动表;
为参加join的字段在被驱动表建立聚集索引,其次是非聚集索引;
尽可能减少join的字段数量,或者使用长度比较小的字段来join,这样Join Buffer一次可以缓存更多条记录。

inner join时,MySQL会自动将小表作为驱动表,大表作为被驱动表。

扫描整张表是成本非常高的操作。

到此这篇关于MySQL join查询的原理的文章就介绍到这了,更多相关MySQL join查询内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • mysql三种批量增加的性能分析

    mysql三种批量增加的性能分析

    最近在深入学习hibernate,在进行批量操作时,发现hibernate批量操作性能非常低.于是就想找一个性能较高的方法,在对jdbc、jdbcTemplate、hibernate进行测试后,发现jdbc的执行效率是最高的,jdbcTemplate也很相近,hibernate就不考虑了,惨不忍睹啊
    2012-08-08
  • Mysql主从三种复制模式(异步复制,半同步复制,组复制)

    Mysql主从三种复制模式(异步复制,半同步复制,组复制)

    这篇文章主要介绍了Mysql主从三种复制模式(异步复制,半同步复制,组复制),MySQL异步复制是主从复制过程中默认的复制模式,下文简单介绍,感兴趣的朋友可以参考一下
    2022-08-08
  • gorm操作MySql数据库的方法

    gorm操作MySql数据库的方法

    这篇文章主要介绍了gorm操作MySql数据库的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-03-03
  • 深入mysql慢查询设置的详解

    深入mysql慢查询设置的详解

    本篇文章是对mysql慢查询设置进行了详细的分析介绍,需要的朋友参考下
    2013-06-06
  • 记一次MySQL更新语句update的踩坑

    记一次MySQL更新语句update的踩坑

    这篇文章主要介绍了记一次MySQL更新语句update的踩坑,帮助大家更好的理解和使用MySQL的更新语句,感兴趣的朋友可以了解下
    2020-11-11
  • MySQL数据库JDBC编程详解流程

    MySQL数据库JDBC编程详解流程

    JDBC是指Java数据库连接,是一种标准Java应用编程接口( JAVA API),用来连接 Java 编程语言和广泛的数据库。从根本上来说,JDBC 是一种规范,它提供了一套完整的接口,允许便携式访问到底层数据库,本篇文章我们来了解MySQL连接JDBC的流程方法
    2022-01-01
  • mysql中ROW_FORMAT的选择问题

    mysql中ROW_FORMAT的选择问题

    这篇文章主要介绍了mysql中ROW_FORMAT的选择问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-10-10
  • MySQL关键字IN与EXISTS的使用与区别详解

    MySQL关键字IN与EXISTS的使用与区别详解

    in和exists是两种常用的条件查询关键字,两种常用于子查询,它们在某些情况下可以互换使用,但它们的工作方式和效率可能会有所不同,这篇文章主要给大家介绍了关于MySQL关键字IN与EXISTS的使用与区别的相关资料,需要的朋友可以参考下
    2024-09-09
  • mysql拆分字符串作为查询条件的示例代码

    mysql拆分字符串作为查询条件的示例代码

    本文主要介绍了mysql拆分字符串作为查询条件的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • mysql密码忘记后如何修改密码(2022年最新版详细教程保姆级)

    mysql密码忘记后如何修改密码(2022年最新版详细教程保姆级)

    因为长时间不操作mysql而忘记root密码的朋友估计不在少数,下面这篇文章主要给大家介绍了关于mysql密码忘记后如何修改密码的相关资料,本教程是2022年最新版详细教程保姆级,需要的朋友可以参考下
    2022-04-04

最新评论