【问题标题】:MySQL - find difference between rows of the same tableMySQL - 查找同一张表的行之间的差异
【发布时间】:2011-03-23 09:11:30
【问题描述】:

我有一个包含带有时间戳的汇总结果的表格 - 这意味着每次的每个结果都是迄今为止的总数:

date       | time  | ip       | result
---------------------------------------
2011-03-01 | 10:00 | 10.0.0.1 | 200
2011-03-01 | 11:00 | 10.0.0.1 | 303
2011-03-01 | 12:00 | 10.0.0.1 | 415
2011-03-01 | 13:00 | 10.0.0.1 | 628
2011-03-01 | 10:00 | 10.0.0.2 | 198
2011-03-01 | 11:00 | 10.0.0.2 | 234
2011-03-01 | 12:00 | 10.0.0.2 | 373
2011-03-01 | 13:00 | 10.0.0.2 | 512

我正在尝试制定一个查询,以获取每个时间范围之间的增量:

date       | time  | ip       | diff
---------------------------------------
2011-03-01 | 10:00 | 10.0.0.1 | 200
2011-03-01 | 11:00 | 10.0.0.1 | 103
2011-03-01 | 12:00 | 10.0.0.1 | 112
2011-03-01 | 13:00 | 10.0.0.1 | 213
2011-03-01 | 10:00 | 10.0.0.2 | 198
2011-03-01 | 11:00 | 10.0.0.2 |  36
2011-03-01 | 12:00 | 10.0.0.2 | 139
2011-03-01 | 13:00 | 10.0.0.2 | 139
...

因此,每个日期/ip 分组的每一行都会减去它之前的一行(或 0)。 有什么简单的方法可以做到这一点吗?谢谢。

【问题讨论】:

  • 您可能想详细说明如何处理日期更改。结果是每天从零开始,还是只在文件开头从零开始?
  • 如果有一个 11:00 的数据点并且没有 10:00、只有 09:00 或没有那一天之前怎么办?

标签: mysql query-help


【解决方案1】:

这是一个没有变量的解决方案。我假设您的初始数据位于名为 thetable 的表中。

SELECT date, time, ip,
    result - IFNULL( (
        SELECT MAX( result ) 
        FROM thetable
        WHERE ip = t1.ip
        AND ( date < t1.date
            OR date = t1.date AND time < t1.time )
    ) , 0) AS diff
FROM thetable AS t1
ORDER BY ip, date, time

在这里,我们通过子选择获取先前的值(来自同一 ip 的先前时间戳的最大 result)。如果这是第一个值,IFNULL 会给我们一个 0,因此初始结果会正确显示。

我还建议将以下索引添加到thetable

CREATE INDEX sort1 ON thetable (ip, date, time);

【讨论】:

  • 好方案,效率更高。
  • 谢谢。 date 字段的处理有一点问题。现在好了。
  • 我也遇到了一些麻烦,变量 ;)
【解决方案2】:

试试这个 -

SET @f_rank = 0;
SET @s_rank = 0;
SET @f_date = NULL;
SET @f_ip = NULL;
SET @s_date = NULL;
SET @s_ip = NULL;
SELECT t1.date, t1.time, t1.ip, IF(t2.result IS NULL, t1.result, t1.result - t2.result) diff FROM
    (SELECT *, IF(@f_date = date AND @f_ip = ip, @f_rank:=@f_rank + 1, @f_rank:=1) AS rank, @f_date := date, @f_ip := ip FROM table1 ORDER BY date, ip, time) t1
  LEFT JOIN 
    (SELECT *, IF(@s_date = date AND @s_ip = ip, @s_rank:=@s_rank + 1, @s_rank:=1) AS rank, @s_date := date, @s_ip := ip FROM table1 ORDER BY date, ip, time) t2
  ON t1.date = t2.date AND t1.ip = t2.ip AND (t1.rank = t2.rank + 1);

+------------+----------+----------+------+
| date       | time     | ip       | diff |
+------------+----------+----------+------+
| 2011-03-01 | 10:00:00 | 10.0.0.1 |  200 |
| 2011-03-01 | 11:00:00 | 10.0.0.1 |  103 |
| 2011-03-01 | 12:00:00 | 10.0.0.1 |  112 |
| 2011-03-01 | 13:00:00 | 10.0.0.1 |  213 |
| 2011-03-01 | 10:00:00 | 10.0.0.2 |  198 |
| 2011-03-01 | 11:00:00 | 10.0.0.2 |   36 |
| 2011-03-01 | 12:00:00 | 10.0.0.2 |  139 |
| 2011-03-01 | 13:00:00 | 10.0.0.2 |  139 |
+------------+----------+----------+------+

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-07-12
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多