9

考虑下表:

SELECT id, value FROM table ORDER BY id ASC;
+-----+---------+
| id  | value   |
+-----+---------+
| 12  | 158     |
| 15  | 346     |
| 27  | 334     |
| 84  | 378     |
| 85  | 546     |
+-----+---------+

id列是自动递增的,但包含间隙。该value列是数字。

我想通过与上面两行相关的设置来查看value随时间的增加。那是我想设置行(546)相对于行(334)的行。因此,要为行计算的值是 546/334=1.63473。valuevalueid=85valueid=85valueid=27id=85

这是我想要达到的结果:

SELECT id, value, ...;
+-----+---------+---------------------+
| id  | value   | value/lag(value, 2) | (the syntax value/lag(value, 2) is made up)
+-----+---------+---------------------+
| 12  | 158     | NULL                |
| 15  | 346     | NULL                |
| 27  | 334     | 2.11392             | (334/158=2.11392)
| 84  | 378     | 1.09248             | (378/346=1.09248)
| 85  | 546     | 1.63473             | (546/334=1.63473)
+-----+---------+---------------------+

如何在 MySQL 中执行这种滞后?

请注意,该id列包含间隙,因此仅在同一个表上加入是t1.id = t2.id - 2行不通的。

4

2 回答 2

9

这是一个在 MySQL 中返回您想要的内容的解决方案

SET @a :=0;
SET @b :=2;
SELECT r.id, r.value, r.value/r2.value AS 'lag'
FROM
(SELECT if(@a, @a:=@a+1, @a:=1) as rownum, id, value FROM results) AS r
LEFT JOIN
(SELECT if(@b, @b:=@b+1, @b:=1) as rownum, id, value FROM results) AS r2
ON r.rownum = r2.rownum

MySQL 5.1 不喜欢针对子查询的自联接,因此您必须计算行两次,因此不像它可能的那样整洁或可扩展,但它确实使指定滞后变得简单。

对于使用 Oracle 的读者来说,这要容易得多

SELECT id, value, value/lag(value, 2) over (order by id) as lag from results;
于 2011-03-30T10:40:45.143 回答
0

由于当前行和要从中获取“历史”数据的行之间只有两行,因此您是否可以使用变量来临时存储数据,例如:

set @oldid0=999999;
set @oldid1=999999;
set @oldid2=999999;
select @oldid0:=@oldid1,@oldid1:=@oldid2,@oldid2:=id, value/@oldid0 from table order by id asc;

这是一个非常不整洁的解决方案,但我认为它会完成这项工作。也许有一些方法可以防止显示变量,我还没有研究那么远。

于 2011-03-30T08:05:03.187 回答