每周运行当前和上一年的总计
Posted
技术标签:
【中文标题】每周运行当前和上一年的总计【英文标题】:Running Total for Current & Previous Year on weekly basis 【发布时间】:2017-02-23 22:30:01 【问题描述】:此查询提供当年和上一年至今的价格和平方英尺的年初至今数字。这更像是当前年份和上一年的运行总计,在这种情况下,从 1 到 7 等周......(2017 年第 7 周于 2017 年 2 月 19 日结束)本年度和上一年(2016 年第 7 周于 2016 年 2 月 22 日结束)。我使用子查询的原因是因为这是我知道的唯一解决这种情况的方法。当然,如果您认为执行此查询有更短、更可行的替代方案,请提出建议。
Actual_Sale_Date
保存一周中所有 7 天的数据,但我们在周日截止,这就是为什么 2/22/2016(周日结束 2016 年第 7 周)和 2/19/2017(周日结束第 7 周) 2017 年)。
我试过"Actual_Sale_Date" = date_trunc('week', now())::date - 1
这个函数只返回在过去的星期日结束的前一周数据。我看了一下interval
,因为dateadd
在postgresql 中不存在,但我无法解决它。
我的查询:
select (money(Sum("Price") / COUNT("Price"))) as "Avg_Value YTD",
Round(Avg("Price"/"Sq_Ft"),+2) as "Avg_PPSF YTD",
(select
(money(Sum("Price") / COUNT("Price"))) from allsalesdata
where "Actual_Sale_Date" >= '01/01/2016' AND "Actual_Sale_Date" < '02/22/2016'
and "Work_ID" = 'SO') AS "Last Year at this time Avg_Value",
(select Round(Avg("Price"/"Sq_Ft"),+2)
from allsalesdata
where "Actual_Sale_Date" >= '01/01/2016' AND "Actual_Sale_Date" < '02/22/2016'
and "Work_ID" = 'SO') AS "Last Year at this time Avg_PPSF"
from allsalesdata
where "Actual_Sale_Date" >= '01/01/2017' AND "Actual_Sale_Date" <'02/20/2017'
and "Work_ID" = 'SO'
样本数据:
Price Sq_Ft Actual_Sale_Date Work_ID
45871 3583 01/15/2016 SO
55874 4457 02/05/2016 SO
88745 4788 02/20/2016 SO
58745 1459 01/10/2016 SO
88749 2145 01/25/2017 SO
74856 1478 01/25/2017 SO
74586 4587 01/31/2017 ABC
74745 1142 02/10/2017 SO
74589 2214 02/19/2017 SO
【问题讨论】:
是否需要按周生成日期系列? 查询有weeks
系列要求。随着年份的推移,列总计。
对不起,我不清楚,按周累计?
例如,Total 1st week 2016 Total 1st week 2017?
是的,你可以这么说。例如。 "Actual_Sale_Date" = date_trunc('week', now())::date - 1
这个函数只返回在过去的星期日结束的前一周数据。我看了一下interval
,因为dateadd
在postgresql 中不存在,但我无法解决它。 For example, Total 1st week 2016 Total 1st week 2017?
是的,但是当我们进入 3 月、4 月等时总计...
【参考方案1】:
这应该是您所需要的(假设您有最新版本的 PG):
SELECT DISTINCT wk AS "Week",
sum("Price")::money FILTER (WHERE yr = 2017) OVER w /
count("Price") FILTER (WHERE yr = 2017) OVER w AS "Avg_Value YTD",
sum("Price")::money FILTER (WHERE yr = 2017) OVER w /
sum("Sq_Ft") FILTER (WHERE yr = 2017) OVER w AS "Avg_PPSF YTD",
sum("Price")::money FILTER (WHERE yr = 2016) OVER w /
count("Price") FILTER (WHERE yr = 2016) OVER w AS "Last Year this time Avg_Value",
sum("Price")::money FILTER (WHERE yr = 2016) OVER w /
sum("Sq_Ft") FILTER (WHERE yr = 2016) OVER w AS "Last Year this time Avg_PPSF",
FROM (
SELECT extract(isoyear from "Actual_Sale_Date")::integer AS yr,
extract(week from "Actual_Sale_Date")::integer AS wk,
"Price", "Sq_Ft"
FROM allsalesdata
WHERE "Work_ID" = 'SO') sub
-- optional, show only completed weeks in this year:
WHERE wk <= extract(week from CURRENT_DATE)::integer - 1
WINDOW w AS (ORDER BY wk)
ORDER BY wk;
在内部查询中,为每次销售提取销售日期的年份和星期。根据您的要求,这一周从星期一开始。
在主查询中,这些行作为单个分区帧处理,即从分区的开始(=第一行)到当前的最后一个节点排。由于窗口定义按wk
对行进行排序,因此从开始(周= 1)到当前周的所有行都包含在摘要中。这将为您提供运行总数。 sum()
和 count()
函数按相关年份过滤,DISTINCT
子句确保您每周只获得一行。
【讨论】:
我使用的是 PostgreSQL 9.6beta3,由 Visual C++ build 1800 编译,64 位。 谢谢帕特里克,它工作得很好,但你知道为什么它显示第八周的数据,即使我们仍处于 2017 年的第八周。它应该等到本周的星期日还是星期一下周? @JakeWagner:此查询显示从年初到当前一周(包括)的数据。这周是第 8 周。 如果只想要整周,可以修改 WHERE 子句并减去 1:WHERE wk <= extract(week from CURRENT_DATE)::integer - 1
。以上是关于每周运行当前和上一年的总计的主要内容,如果未能解决你的问题,请参考以下文章