Postgres:找到时间戳行之间的最小差异

时间:2015-09-08 16:15:05

标签: postgresql function timestamp

我有用户,他们跑步,我有每圈的时间戳记录。

我想以最快圈速排序用户。

这个想法是使用postgres的“滞后”功能,使用“GROUP BY user_id”

我尝试了一些东西,但我不知道“滞后”,而且我无法获得和窗口功能(圈)使用聚合功能(分钟)

http://sqlfiddle.com/#!15/ec9dc/6

你有什么想法吗?

由于

1 个答案:

答案 0 :(得分:2)

如果您想按照他们的单圈持续时间对用户进行排序,则必须按行user_id对行进行分区。否则,不同用户的单圈时间将会混合。

我已经采用了你的示例代码并对其进行了一些修改:我删除了一个时间戳,插入了两次,然后我引入了一个包含窗口函数rank()lag()的视图。前者用于计算每个用户的圈数,后者用于确定当前时间戳的前一个时间戳。

BEGIN;

CREATE TABLE lap
(
  user_id text,
  timestamp timestamp without time zone
);

INSERT INTO lap VALUES
    ('a', '2015-08-20 16:14:30.568'),
    ('a', '2015-08-20 16:16:13.06'),
    ('b', '2015-08-20 16:16:18.06'),
    ('b', '2015-08-20 16:16:25.63'),
    ('b', '2015-08-20 16:17:10.568'),
    ('a', '2015-08-20 16:17:25.63'),
  --('a', '2015-08-20 16:17:34.087'), -- Timestamp was inserted twice.
    ('a', '2015-08-20 16:17:34.087');

CREATE OR REPLACE VIEW user_lap_duration AS
    SELECT
        user_id,
        -1+rank() OVER w AS lap_nr,
        lag(timestamp) OVER w AS timestamp_prev,
        timestamp,
        timestamp - lag(timestamp)
            OVER w
          AS lap_duration
    FROM
        lap
        WINDOW w AS (PARTITION BY user_id ORDER BY timestamp);

SELECT
    user_id,
    lap_nr,
    lap_duration
FROM
    user_lap_duration
WHERE
    lap_nr <> 0
ORDER BY
    lap_duration;

ROLLBACK;

在代码上运行会产生以下输出。

 user_id | lap_nr | lap_duration 
---------+--------+--------------
 b       |      1 | 00:00:07.57
 a       |      3 | 00:00:08.457
 b       |      2 | 00:00:44.938
 a       |      2 | 00:01:12.57
 a       |      1 | 00:01:42.492
(5 rows)