zoukankan      html  css  js  c++  java
  • MySQL -- STRAIGHT_JOIN

    STRAIGHT_JOIN,在数据量大的联表查询中灵活运用的话,能大大缩短查询时间。

    首先来解释下STRAIGHT_JOIN到底是用做什么的:

    -- STRAIGHT_JOIN is similar to JOIN, except that the left table is always read before the right table.  This can be used for those (few) cases for which the join optimizer puts the tables in the wrong order.

    大意是说:说STRAIGHT_JOIN功能同join类似,但能让左边的表来驱动右边的表,能改表优化器对于联表查询的执行顺序。

    下面举例进行大致分析:

    select t1.*
    from Table1 t1
    inner join Table2 t2
    on t1.CommonID = t2.CommonID
    where t1.FilterID = 1

    以上sql大数据量下执行需要30s,是不是很奇怪?明明Table1表的FilterID字段建了索引啊,Table1和Table2的CommonID也建了索引啊。通过explain来分析,你会发现执行计划中表的执行顺序是Table2->Table1。这个时候要略微介绍下驱动表的概念,mysql中指定了连接条件时,满足查询条件的记录行数少的表为驱动表;如未指定查询条件,则扫描行数少的为驱动表。mysql优化器就是这么粗暴以小表驱动大表的方式来决定执行顺序的

    但如下sql的执行时间都少于1s:

    select t1.*
    from Table1 t1
    where t1.FilterID = 1

    select t1.*
    from Table1 t1
    inner join Table2 t2
    on t1.CommonID = t2.CommonID

    这个时候STRAIGHT_JOIN就派上用场,我们对sql进行改造如下:

    select t1.*
    from Table1 t1
    STRAIGHT_JOIN  Table2 t2
    on t1.CommonID = t2.CommonID
    where t1.FilterID = 1

    用explain进行分析,发现执行顺序为Table1->Table2,这时就由Table1来作为驱动表了,Table1中相应的索引也就用上了,执行时间竟然低于1s了。

    分析到这里,必须要重点说下:

    • STRAIGHT_JOIN只适用于inner join,并不使用与left join,right join。(因为left join,right join已经代表指定了表的执行顺序)
    • 尽可能让优化器去判断,因为大部分情况下mysql优化器是比人要聪明的。使用STRAIGHT_JOIN一定要慎重,因为啊部分情况下认为指定的执行顺序并不一定会比优化引擎要靠谱。

    STRAIGHT_JOIN:这个提示可以放置在 SELECT语句的SELECT关键字之后, 也可以放置在任何两个关联表的名字之间。 第一个用法是让查询中的所有表按照在语句中出现的顺序进行关联。第二个用法是固定其前后两个表的关联顺序。  当MySQL没有选择正确的关联顺序的时候,或者由于可能的顺序太多导致MySQL无法评估所有的关联顺序的时候, STRAIGHT_JOIN 都会很有用。 在后面的情况中,MySQL可能会花费大量时间在 “statistics”状态,加上这个提示则会大大减少优化器的搜索时间。

    参考链接: 

    https://www.cnblogs.com/heyonggang/p/9462242.html

    https://blog.huoding.com/2013/06/04/261

  • 相关阅读:
    Bootstrap
    格式化字符串
    闭包函数与装饰器
    正则表达式
    jQuery
    分布式-锁-1.1 多线程锁无法满足的场景
    effective python 读书笔记-第22条: 尽量用辅助类来维护程序的状态,而不要用字典
    effective python 读书笔记:第21条-用只能以关键字形式指定的参数来确保代码明晰
    effective python 读书笔记:第20条-用None和文档字符串来描述具有动态默认值的参数
    git如何将上游(upstream)新建分支(origin没有)导入到origin中?
  • 原文地址:https://www.cnblogs.com/neozheng/p/12915546.html
Copyright © 2011-2022 走看看