zoukankan      html  css  js  c++  java
  • SQL执行顺序

    看到一篇讲sql语句执行顺序的文章,觉得比较有意思,自己测试之后和大家分享一下
    让我们通过一个有趣的统计,来看看SQL语句的执行顺序。
    正好有一些数据资源,那选取大家应该会比较有兴趣的话题,哪个星座的空姐最多?
    选取的表CREW_V
    因为一般一个星座的开始都是在当月的20到23号,因为在做统计的时候,我在每个出生日期-22天,这样,就可能通过月份来代表星座
    例如处女座是在8月23号到9月23号,那-22天后,只要统计出来生日在8月的,就可以粗略的认为这个人是处女座
     
    看下面的几段SQL语句:
    --统计空姐的出生月份
    SELECT TO_CHAR(BIRTH_DT-22, 'MM' ), COUNT(TO_CHAR(BIRTH_DT-22'MM' )) AS BIRTH
      FROM CREW_V
      WHERE SEX = 'F'
       AND NON_CREW_IND = 'N'
      GROUP BY BIRTH
    --统计空姐的出生月份,当月出生人数大于100的才记录
    SELECT TO_CHAR(BIRTH_DT-22, 'MM' ),
           COUNT (TO_CHAR(BIRTH_DT-22, 'MM' )) AS BIRTH
      FROM CREW_V
      WHERE SEX = 'F' AND NON_CREW_IND = 'N'
      GROUP BY TO_CHAR(BIRTH_DT-22, 'MM' )
    HAVING COUNT (BIRTH) > 100
     
    --按照月份出生的人数排序
    SELECT TO_CHAR(BIRTH_DT-22, 'MM' ), COUNT(TO_CHAR(BIRTH_DT-22, 'MM')) AS BIRTH
      FROM CREW_V
     WHERE SEX = 'F'
       AND NON_CREW_IND = 'N'
     GROUP BY TO_CHAR(BIRTH_DT-22, 'MM' )
     ORDER BY BIRTH
     
     
     
    你觉得哪一个能够成功执行?
     

    示例一

    SELECT TO_CHAR(BIRTH_DT, 'MM' ), COUNT(TO_CHAR(BIRTH_DT, 'MM' )) AS BIRTH
      FROM CREW_V
      WHERE SEX = 'F'
       AND NON_CREW_IND = 'N'
      GROUP BY BIRTH
    实际执行顺序:
    1、FROM CREW_V
    2、WHERE SEX = 'F' AND NON_CREW_IND = 'N'
    3、GROUP BY BIRTH
    4、SELECT TO_CHAR(BIRTH_DT, 'MM' ), COUNT(TO_CHAR(BIRTH_DT, 'MM' )) AS BIRTH
     
    很明显,执行GROUP BY BIRTH时别名BIRTH还没有创建,因此它是不能执行成功的。
     
    示例二
    SELECT TO_CHAR(BIRTH_DT, 'MM' ),
           COUNT (TO_CHAR(BIRTH_DT, 'MM' )) AS BIRTH
      FROM CREW_V
      WHERE SEX = 'F' AND NON_CREW_IND = 'N'
      GROUP BY TO_CHAR(BIRTH_DT, 'MM' )
    HAVING COUNT (BIRTH) > 100
     
    这是非常基础的分组查询。但它不能执行成功,因为HAVING的执行顺序在SELECT之上。
    实际执行顺序如下:
    1. FROM CREW_V
    2.  WHERE SEX = 'F' AND NON_CREW_IND = 'N'
    3. GROUP BY TO_CHAR(BIRTH_DT, 'MM' )
    4. HAVING COUNT (BIRTH) > 100
    5. SELECT TO_CHAR(BIRTH_DT, 'MM' ), COUNT (TO_CHAR(BIRTH_DT, 'MM' )) AS BIRTH
    很明显,BIRTH是在最后一句SELECT TO_CHAR(BIRTH_DT, 'MM' ), COUNT (TO_CHAR(BIRTH_DT, 'MM' )) AS BIRTH执行过后生成的新别名。
    因此,在HAVING COUNT (BIRTH) > 100执行时是不能识别BIRTH的。
     
    示例三
    SELECT TO_CHAR(BIRTH_DT, 'MM' ), COUNT(TO_CHAR(BIRTH_DT, 'MM')) AS BIRTH
      FROM CREW_V
     WHERE SEX = 'F'
       AND NON_CREW_IND = 'N'
     GROUP BY TO_CHAR(BIRTH_DT, 'MM' )
     ORDER BY BIRTH
    这一次没有任何问题,能够成功执行。
    1、 FROM CREW_V
    2、 WHERE SEX = 'F'  AND NON_CREW_IND = 'N'
    3、 GROUP BY TO_CHAR(BIRTH_DT, 'MM' )
    4、 SELECT TO_CHAR(BIRTH_DT, 'MM' ), COUNT(TO_CHAR(BIRTH_DT, 'MM')) AS BIRTH
    5、 ORDER BY BIRTH
     

    下面是SELECT语句的逻辑执行顺序:

    1. FROM
    2. ON
    3. JOIN
    4. WHERE
    5. GROUP BY
    6. WITH CUBE or WITH ROLLUP
    7. HAVING
    8. SELECT
    9. DISTINCT
    10. ORDER BY
    11. TOP

    下面给大家分享一点私密的数据

  • 相关阅读:
    spring
    google-c-style
    process想停就停,真爽
    mytop
    Java 图片设置圆角(设置边框,旁白)
    当setTimeout遇到闭包
    FreeMarker辅助
    ImageIO.wtrie生成jpg图片质量损失方案:BufferedImage生成jpg图片文件流
    从BufferedImage到InputStream,实现绘图后进行下载(生成二维码图片并下载)
    使用Javascript 实现类
  • 原文地址:https://www.cnblogs.com/pfxiong/p/3745421.html
Copyright © 2011-2022 走看看