基于日期时间值的相同表上的高效sql子查询

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了基于日期时间值的相同表上的高效sql子查询相关的知识,希望对你有一定的参考价值。

下面我有一个简单的查询来获取今天加入“事件”表和“电影”表的所有电影评级。

Select e.*, m.moviename 
From Event e, movie m 
Where e.eventdate >= DATEADD(day, -1, GETDATE()) 
  and e.moviekey = m.moviekey 
order by e.Ratings desc;

在上面的示例中,您将如何从1周前和1个月前检索评级。因此查询将返回2个额外的列RatingOneMonthAgo,RatingsOneWeekAgo等。

我查看了子查询,并没有点击任何帮助将不胜感激。

谢谢

答案

您可以使用CTEs来提取此信息(类似于使用子查询)。

以下查询假设您每天都有评分,并且没有重复项(同一天的同一部电影的多个评级):

WITH cteOneWeekAgo
AS
(
    SELECT
        moviekey
        , Ratings
    FROM Event
    WHERE CAST(eventdate AS date) = DATEADD(WEEK, -1, CAST(GETDATE() AS date))
)
,

cteOneMonthAgo
AS
(
    SELECT
        moviekey
        , Ratings
    FROM Event
    WHERE CAST(eventdate AS date) = DATEADD(MONTH, -1, CAST(GETDATE() AS date))
)

SELECT
    e.*
    , m.moviename
    , w.Ratings Ratings_OneWeekAgo
    , mth.Ratings Ratings_OneMonthAgo
FROM
    Event e
    JOIN movie m ON e.moviekey = m.moviekey
    LEFT JOIN cteOneWeekAgo w ON e.moviekey = w.moviekey
    LEFT JOIN cteOneMonthAgo mth ON e.moviekey = mth.moviekey
WHERE e.eventdate >= DATEADD(DAY, -1, GETDATE())
ORDER BY e.Ratings DESC

我还写了一个更复杂的查询,如果不存在该日期的评级,它将在您查找的日期之前提取电影的最新评级。

WITH cteOneWeekAgo
AS
(
    SELECT
        moviekey
        , Ratings
        , eventdate
    FROM
        (
            SELECT
                moviekey
                , Ratings
                , eventdate
                , ROW_NUMBER() OVER (PARTITION BY moviekey ORDER BY eventdate DESC) R
            FROM Event
            WHERE CAST(eventdate AS date) <= DATEADD(WEEK, -1, CAST(GETDATE() AS date))
        ) Q
    WHERE R = 1
)
,

cteOneMonthAgo
AS
(
    SELECT
        moviekey
        , Ratings
        , eventdate
    FROM
        (
            SELECT
                moviekey
                , Ratings
                , eventdate
                , ROW_NUMBER() OVER (PARTITION BY moviekey ORDER BY eventdate DESC) R
            FROM Event
            WHERE CAST(eventdate AS date) <= DATEADD(MONTH, -1, CAST(GETDATE() AS date))
        ) Q
    WHERE R = 1
)

SELECT
    e.*
    , m.moviename
    , w.eventdate Ratings_OneWeekAgo_MostRecentDate
    , w.Ratings Ratings_OneWeekAgo
    , mth.eventdate Ratings_OneMonthAgo_MostRecentDate
    , mth.Ratings Ratings_OneMonthAgo
FROM
    Event e
    JOIN movie m ON e.moviekey = m.moviekey
    LEFT JOIN cteOneWeekAgo w ON e.moviekey = w.moviekey
    LEFT JOIN cteOneMonthAgo mth ON e.moviekey = mth.moviekey
WHERE e.eventdate >= DATEADD(DAY, -1, GETDATE())
ORDER BY e.Ratings DESC

以上是关于基于日期时间值的相同表上的高效sql子查询的主要内容,如果未能解决你的问题,请参考以下文章

如何使用具有多个 GROUP BY、子查询和 WHERE IN 在大表上的查询来优化查询?

使用 2GB+ 加速单个表上的 SQL 查询

使用 Oracle SQL 的子查询

获取表上相同值的计数

连接表上的高效 GraphQL 查询

高效的 ORDER BY 与大型表上的依赖子查询