我的Postgresql 9.1数据库中有下表:

select * from ro;
date       |  shop_id | amount 
-----------+----------+--------
2013-02-07 |     1001 |      3
2013-01-31 |     1001 |      2
2013-01-24 |     1001 |      1
2013-01-17 |     1001 |      5
2013-02-10 |     1001 |     10
2013-02-03 |     1001 |      4
2012-12-27 |     1001 |      6
2012-12-20 |     1001 |      8
2012-12-13 |     1001 |      4
2012-12-06 |     1001 |      3
2012-10-29 |     1001 |      3

我试图得到一个移动平均值,将过go 三个星期四的数据进行比较,不包括当前的星期四.我的问题是:

select date, shop_id, amount, extract(dow from date),
avg(amount) OVER (PARTITION BY extract(dow from date) ORDER BY date DESC
                      ROWS BETWEEN 0 PRECEDING AND 2 FOLLOWING)                          
from ro
where extract(dow from date) = 4

这是给出的结果

date       |  shop_id | amount | date_part |        avg         
-----------+----------+--------+-----------+--------------------
2013-02-07 |     1001 |      3 |         4 | 2.0000000000000000
2013-01-31 |     1001 |      2 |         4 | 2.6666666666666667
2013-01-24 |     1001 |      1 |         4 | 4.0000000000000000
2013-01-17 |     1001 |      5 |         4 | 6.3333333333333333
2012-12-27 |     1001 |      6 |         4 | 6.0000000000000000
2012-12-20 |     1001 |      8 |         4 | 5.0000000000000000
2012-12-13 |     1001 |      4 |         4 | 3.5000000000000000
2012-12-06 |     1001 |      3 |         4 | 3.0000000000000000

我想

date       |  shop_id | amount | date_part |        avg         
-----------+----------+--------+-----------+--------------------
2013-02-07 |     1001 |      3 |         4 | 2.6666666666666667
2013-01-31 |     1001 |      2 |         4 | 4.0000000000000000
2013-01-24 |     1001 |      1 |         4 | 6.3333333333333333
2013-01-17 |     1001 |      5 |         4 | 6.0000000000000000
2012-12-27 |     1001 |      6 |         4 | 5.0000000000000000
2012-12-20 |     1001 |      8 |         4 |
2012-12-13 |     1001 |      4 |         4 |
2012-12-06 |     1001 |      3 |         4 |

推荐答案

SQL Fiddle

select
    "date",
    shop_id,
    amount,
    extract(dow from date),
    case when
        row_number() over (order by date) > 3
        then
            avg(amount) OVER (
                ORDER BY date DESC
                ROWS BETWEEN 1 following AND 3 FOLLOWING
            )
        else null end
from (
    select *
    from ro
    where extract(dow from date) = 4
) s

OP查询的问题在于框架规格:

ROWS BETWEEN 0 PRECEDING AND 2 FOLLOWING

除此之外,我的查询通过在应用昂贵的窗口函数之前过滤周四来避免不必要的计算.

如果需要按shop_id进行分区,那么显然要将partition by shop_id添加到两个函数avgrow_number中.

Postgresql相关问答推荐

在输入稍有错误的PostgreSQL表中进行快速字符串搜索

PostgreSQL散列连接与嵌套循环和散列索引

如何在postgres中测试锁定

在 Windows 上使用 rsync 进行 Postgresql 副本升级

是否可以短时间运行 VACUUM FULL 并获得一些好处?

在 postgresql 中查找表的所有依赖项

如何在 PostgreSQL 中使用正则表达式删除单词之间的所有特殊字符和多余空格

pgadmin db 限制服务器属性不起作用

Postgres 会话处于空闲状态,query = COMMIT 或 ROLLBACK

在 to_tsquery 中转义特殊字符

Postgres COPY FROM csv file-No such file or directory

从局域网访问 PostgreSQL 服务器

PostgreSQL:将时间戳转换为时间 - 或仅从时间戳列中检索时间

从 PostgreSQL 中的时间戳获取日期

与 Oracle 的 CONNECT BY ... START WITH 等效的 PostgreSQL 语法是什么?

在 MAC OS X 10.6 for PostgreSQL 上设置 SHMMAX 等值

Postgres 数据库文件保存在 ubuntu 中的什么位置?

什么是 Postgres session?

如何使 array_agg() 像 mySQL 中的 group_concat() 一样工作

如何在创建数据库时安装 Postgres 扩展?