我有用户,他们跑步,我有每圈的时间戳记录。
我想以最快圈速排序用户。
这个想法是使用postgres的“滞后”功能,使用“GROUP BY user_id”
我尝试了一些东西,但我不知道“滞后”,而且我无法获得和窗口功能(圈)使用聚合功能(分钟)
http://sqlfiddle.com/#!15/ec9dc/6
你有什么想法吗?
由于
答案 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)