重写查询以使用除 CTE 和子查询之外的窗口函数

Posted

技术标签:

【中文标题】重写查询以使用除 CTE 和子查询之外的窗口函数【英文标题】:Rewrite query to use window functions except CTE's and subqueries 【发布时间】:2018-01-22 06:24:46 【问题描述】:

我有 2 个相关的表。

package:

| id_sort | volume | date |
---------------------------
| int     | float  | date |

sort:

| id  | id_standard |
---------------------
| int | int         |

我需要计算某种类型的音量占sort.id_standard 过滤的总音量的百分比。有 CTE 的完整查询

WITH total(volume) AS (
    SELECT SUM(package.volume) as volume
    FROM package
    LEFT JOIN sort ON sort.id = package.id_sort
    WHERE
        sort.id_standard IN (2,3)
        AND package.date >= CONVERT(VARCHAR(7), GETDATE(), 120) + '-01 08:00:00' -- Start of current month
        AND package.id_conv <> 12 -- additional filter
), filtered(volume) AS (
    SELECT SUM(package.volume) as volume
    FROM package
    WHERE package.id_sort = 17
        AND package.date >= CONVERT(VARCHAR(7), GETDATE(), 120) + '-01 08:00:00' -- start of current month
        AND package.id_conv <> 12 -- additional filter
)
SELECT CAST((filtered.volume * 100 / total.volume) AS NUMERIC(3,2)) [percentage] FROM total, filtered;

我敢肯定,窗口函数在那里会做得最好,但在现实生活中没有使用它们的经验。

【问题讨论】:

您如何确定窗口函数在此处工作得最好?如果您使用窗口函数,结果中将返回更多行并且可能不正确。 【参考方案1】:

这里不需要使用window 函数,这样的东西应该会有所帮助

SELECT [percentage] = Cast(( volume * 100 / Nullif(volume,0) ) AS NUMERIC(3, 2))
FROM   (SELECT total = Sum(volume),
               volume = Sum(CASE WHEN p.id_sort = 17 THEN p.volume ELSE 0 END)
        FROM   package p
               LEFT JOIN sort s
                      ON s.id = p.id_sort
                         AND sort.id_standard IN ( 2, 3 )
        WHERE  p.date >= CONVERT(VARCHAR(7), Getdate(), 120)
                         + '-01 08:00:00' -- Start of current month
               AND p.id_conv <> 12)a 

【讨论】:

需要将 id_standard 的过滤器替换为连接条件中的常见条件(有 LEFT JOIN,所以我确实有错误的结果集没有被该条件过滤)

以上是关于重写查询以使用除 CTE 和子查询之外的窗口函数的主要内容,如果未能解决你的问题,请参考以下文章

带有子查询的 CTE 查询在小型索引表上很慢;如何在 MySQL 上进行优化?

MySQL 8与MariaDB:两者窗口函数和CTE的比较

重写查询以进行分页

css 媒体查询以针对除 IE 之外的所有浏览器

PostgREST 在子查询或 CTE 中使用限制和偏移量

需要 SQL 查询以获取除 sysadmin 程序之外的所有并发程序的列表