簡體   English   中英

提高 MySQL 左連接子查詢的性能

[英]Improving the performance of a MySQL left join sub query

我有以下 MySQL 查詢,它計算給定日期范圍內每個月的訂單總數,例如一年。 查詢正常工作,但性能很慢(大約 250 毫秒)。

關於如何重寫它以提高效率的任何想法?

WITH recursive `dates` AS (
    (
        SELECT '2019-11-28' AS item
    )
    UNION
    ALL (
        SELECT
            item + INTERVAL 1 DAY
        FROM
            `dates`
        WHERE
            item + INTERVAL 1 DAY <= '2020-11-27'
    )
)
SELECT
    DATE_FORMAT(`item`, '%b %y') AS `date`,
    COUNT(`orders`.`id`) AS `total`
FROM
    `dates`
    LEFT JOIN (
        SELECT
            `orders`.`id`,
            `orders`.`created_at`
        FROM
            `orders`
            INNER JOIN `locations` ON `orders`.`location_id` = `locations`.`id`
        WHERE
            `orders`.`shop_id` = 10379184
            AND `locations`.`country_id` = 128
            AND `orders`.`created_at` >= '2019-11-28 12:01:42'
            AND `orders`.`created_at` <= '2020-11-27 12:01:42'
    ) AS `orders` ON DATE(`orders`.`created_at`) = `dates`.`item`
GROUP BY
    `date`

更新:有些人建議使用兩個左連接,但是如果我這樣做,則不會應用country_id過濾器:

WITH recursive `dates` AS (
    (
        SELECT
            '2019-11-28' AS item
    )
    UNION
    ALL (
        SELECT
            item + INTERVAL 1 DAY
        FROM
            `dates`
        WHERE
            item + INTERVAL 1 DAY <= '2020-11-27'
    )
)
SELECT
    DATE_FORMAT(`item`, '%b %y') AS `date`,
    COUNT(`orders`.`id`) AS `total`
FROM
    `dates`
    LEFT JOIN `orders` USE INDEX (`orders_created_at_index`) ON DATE(`created_at`) = `dates`.`item`
    AND `orders`.`shop_id` = 10379184
    AND `orders`.`created_at` >= '2019-11-28 12:22:43'
    AND `orders`.`created_at` <= '2020-11-27 12:22:43'
    LEFT JOIN `locations` ON `orders`.`location_id` = `locations`.`id`
    AND `locations`.`country_id` = 128
GROUP BY
    `date`

謝謝!

我建議使用相關子查詢:

SELECT DATE_FORMAT(d.item, '%b %y') AS `date`,
       (SELECT COUNT(*)
        FROM orders o JOIN
             locations l
             ON o.location_id = l.id
        WHERE shop_id = 10379184 AND
              country_id = 128 AND
              o.created_at >= d.item AND
              o.created_at < d.item + interval 1 day
       ) as total
FROM dates d;

這避免了外部聚合,這通常是一種性能改進。

此外,索引可能有助於查詢,但不清楚country_idshop_id等列的來源。

經過多次修改,我制作了以下運行時間不到 40 毫秒的代碼,這足以滿足我的需求。 我仍然認為這並不理想,歡迎任何改進......

SELECT
    `date`,
    COUNT(`order`)
FROM
    (
        WITH recursive `dates` AS (
            (
                SELECT
                    '2019-11-28' AS item
            )
            UNION
            ALL (
                SELECT
                    item + INTERVAL 1 DAY
                FROM
                    `dates`
                WHERE
                    item + INTERVAL 1 DAY <= '2020-11-27'
            )
        )
        SELECT
            DATE_FORMAT(`item`, '%b %y') AS `DATE`,
            `orders`.`id` AS `order`,
            `locations`.`id` AS `location`
        FROM
            `dates`
        LEFT JOIN 
            `orders` 
        ON 
            DATE(`created_at`) = `dates`.`item`
        AND 
            `orders`.`shop_id` = 10379184
        AND 
            `orders`.`created_at` >= '2019-11-28 12:22:43'
        AND 
            `orders`.`created_at` <= '2020-11-27 12:22:43'
        LEFT JOIN 
            `locations` 
        ON 
            `orders`.`location_id` = `locations`.`id`
        AND 
            `locations`.`country_id` = 209
    ) AS items
WHERE
    (
        `order` IS NULL
        AND `location` IS NULL
    )
    OR (
        `order` IS NOT NULL
        AND `location` IS NOT NULL
    )
GROUP BY
    `date`

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM