Mysql组由记录X之间的delta组成

时间:2015-04-14 13:24:29

标签: php mysql group-by delta

我需要帮助来编写MySQL查询。

我的表中有一些日志,其中一列是unix时间戳。 我想对这些记录进行分组(GROUP BY),以便在每个记录之间的近距离时间(即5秒)内进行的事件都在一个组中。

例如:

表:

timestamp
----------
1429016966
1429016964
1429016963
1429016960
1429016958
1429016957
1429016950
1429016949
1429016943
1429016941
1429016940
1429016938

成为这样的团体:

GROUP_CONCAT(timestamp)                                            | COUNT(*)
----------------------------------------------------------------------------- 
1429016966,1429016964,1429016963,1429016960,1429016958,1429016957  |    6
1429016950,1429016949                                              |    2                         
1429016943,1429016941,1429016940,1429016938                        |    4

当然我可以在php中使用数据数组,但我认为mysql可以更快地完成。

2 个答案:

答案 0 :(得分:0)

我开始使用变量来获取每一行的位置,其中1是最高时间列,最后是最低值,如下所示:

SET @a := 0;

SELECT timeCol, @a := @a + 1 AS position
FROM myTable
ORDER BY timeCol DESC;

为简单起见,我们将调用此positionsTable,以便查询的其余部分更具可读性。创建该表后,我使用了'time_group'变量来检查前一行是否在最后5秒内。如果是,我们保持相同的time_group。这听起来很丑陋,看起来很丑陋,但就像这样:

SELECT m.timeCol, m.position,
  CASE WHEN (SELECT p.timeCol FROM positionsTable p WHERE p.position = m.position - 1) <= m.timeCol + 5 
    THEN @time_group 
    ELSE @time_group := @time_group + 1 END AS timeGroup
FROM positionsTable m;

最后,使用它作为子查询,您可以将它们分组:

SELECT GROUP_CONCAT(timeCol), COUNT(*)
FROM(
  SELECT m.timeCol, m.position,
    CASE WHEN (SELECT p.timeCol FROM positionsTable p WHERE p.position = m.position - 1) <= m.timeCol + 5 
    THEN @time_group 
    ELSE @time_group := @time_group + 1 END AS timeGroup
  FROM positionsTable m) tmp
GROUP BY timeGroup;

以下是SQL Fiddle示例。

答案 1 :(得分:0)

http://sqlfiddle.com/#!9/37d88/20

SELECT GROUP_CONCAT(t1.t) as `time`,
     COUNT(*)
  FROM (SELECT * 
        FROM  table1 
        ORDER BY t) as t1
GROUP BY CASE WHEN (@start+5)>=t THEN @start 
   ELSE @start:=t END