Warning: file_get_contents(/data/phpspider/zhask/data//catemap/8/mysql/70.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Php 获取两行之间的差值,并对差值执行数学运算_Php_Mysql_Sql - Fatal编程技术网

Php 获取两行之间的差值,并对差值执行数学运算

Php 获取两行之间的差值,并对差值执行数学运算,php,mysql,sql,Php,Mysql,Sql,例如,我的数据集如下所示 ID|Value 1|10 1|3 2|9 2|10 2|15 我需要找到相同ID的两个连续元素之间的差异并存储该值,然后再次找到差异并将存储的值与新的差异相乘所有具有相同ID的值。这必须继续,直到有具有相同ID的元素。 因此,我的最终数据集应该是 ID|Value 1|-7 2|5 有人建议我如何使用sql计算连续行之间的差异吗- MariaDB [sandbox]> SELECT T.*, -> IF(T.TID <>

例如,我的数据集如下所示

ID|Value
 1|10
 1|3
 2|9
 2|10
 2|15
我需要找到相同ID的两个连续元素之间的差异并存储该值,然后再次找到差异并将存储的值与新的差异相乘所有具有相同ID的值。这必须继续,直到有具有相同ID的元素。 因此,我的最终数据集应该是

ID|Value
 1|-7
 2|5

有人建议我如何使用sql计算连续行之间的差异吗-

MariaDB [sandbox]> SELECT T.*,
    -> IF(T.TID <> @P ,@RN:=1,@RN:=@RN+1) RN,
    -> IF(T.TID <> @P ,0, @PVAL) PVAL,
    -> IF(T.TID <> @P ,0, T.VALUE * 1.00 - @PVAL) DIFF,
    -> @P:=T.TID,
    -> @PVAL:=T.VALUE
    -> FROM(SELECT @RN:=0,@P:=0,@PVAL:=0.00,@DIFF:=0.00) RN,T
    -> ORDER BY T.TID,T.dt;
+------------+------+-------+------+------+------+-----------+----------------+
| dt         | TID  | Value | RN   | PVAL | DIFF | @P:=T.TID | @PVAL:=T.VALUE |
+------------+------+-------+------+------+------+-----------+----------------+
| 2017-01-01 |    1 |    10 |    1 | 0    |    0 |         1 |             10 |
| 2017-01-02 |    1 |     3 |    2 | 10   |   -7 |         1 |              3 |
| 2017-01-01 |    2 |     9 |    1 | 0    |    0 |         2 |              9 |
| 2017-01-02 |    2 |    10 |    2 | 9    |    1 |         2 |             10 |
| 2017-01-03 |    2 |    15 |    3 | 10   |    5 |         2 |             15 |
+------------+------+-------+------+------+------+-----------+----------------+
5 rows in set (0.00 sec)
但是,据我所知,sql中没有任何一个版本的聚合函数可以计算产品,如果这个词正确的话。更重要的是,你似乎想放弃每一个tid的第一行。 您可以通过组将差异连接到一个表中,然后调用一个存储过程来启动对动态sql的调用来实现这一点。顺便说一下,您不能在存储函数中使用动态sql

这个存储过程

drop procedure if exists f;
delimiter //

CREATE DEFINER=`root`@`localhost` procedure `F`(
    instring varchar(200)
)

LANGUAGE SQL
NOT DETERMINISTIC
CONTAINS SQL
SQL SECURITY DEFINER
COMMENT ''
begin
declare   tempstring varchar(100);
declare   outstring  varchar(100);
declare  tempid int;
declare  checkit int;
declare maxrecs int;
set checkit = 0;
set maxrecs = (select count(*) from temp);

looper: while   checkit < maxrecs do
select tid,calc into    tempid,tempstring from temp limit checkit,1; 
set checkit = checkit + 1;  
if instr(tempstring,',')  = 0 then 
    update temp
    set    rsult  = tempstring; 
else
    set outstring  = replace(tempstring,',','*');
    set outstring  = concat('update temp set rsult = (Select ',outstring,') where tid = ', tempid, ';');
    set @sqlstmt = outstring;
    #select checkit, maxrecs, outstring;
    prepare sqlstmt from @sqlstmt;  
    execute sqlstmt;
    deallocate prepare sqlstmt;
end if;

end while;
end //

delimiter ;
在此脚本中调用时,将生成所需的结果

*ariaDB [sandbox]> drop table if exists temp;
Query OK, 0 rows affected (0.09 sec)

MariaDB [sandbox]> create table temp as
    -> select s.tid,group_concat(s.diff) calc, 1.00 as rsult
    -> from
    -> (
    -> SELECT T.*,
    -> IF(T.TID <> @P ,@RN:=1,@RN:=@RN+1) RN,
    -> IF(T.TID <> @P ,0, @PVAL) PVAL,
    -> IF(T.TID <> @P ,0, T.VALUE * 1.00 - @PVAL) DIFF,
    -> @P:=T.TID,
    -> @PVAL:=T.VALUE
    -> FROM(SELECT @RN:=0,@P:=0,@PVAL:=0.00,@DIFF:=0.00) RN,T
    -> ORDER BY T.TID,T.dt
    -> ) s
    -> where s.rn <> 1
    -> group by s.tid;
Query OK, 2 rows affected (0.23 sec)
Records: 2  Duplicates: 0  Warnings: 0

MariaDB [sandbox]>
MariaDB [sandbox]>
MariaDB [sandbox]> call f('1');
Query OK, 0 rows affected (0.07 sec)

MariaDB [sandbox]>
MariaDB [sandbox]> select * from temp;
+------+-----------+-------+
| tid  | calc      | rsult |
+------+-----------+-------+
|    1 | -7.00     | -7.00 |
|    2 | 1.00,5.00 |  5.00 |
+------+-----------+-------+
2 rows in set (0.00 sec)*

SQL表表示无序集。因此,不存在两个连续的[行],除非有一列指定顺序。我看不到这样的列。id 2的-5从何而来?@Barmar抱歉,它是5如果您有另一个未显示的列指定顺序,您可以使用用户定义的变量来保存前一行的值,然后从中减去当前行的值。@AbuKurian为什么是10-9而不是9-10?ID 1是10-3。