14

我想编写一个查询,它返回所有行,直到其中一个列值的总和达到某个值。

例如在下表中:

           DATE             ETC      Meeting
    2013-02-01 00:00:00    85482        1
    2013-02-01 00:00:00    47228        2
    2013-02-02 00:00:00    12026        4
    2013-02-03 00:00:00    78927        6
    2013-02-04 00:00:00    85662        2
    2013-03-05 00:00:00    47978        1
    2013-08-07 00:00:00     8582        1

如果我想获取行,直到列的总和Meeting等于 7。

           DATE             ETC      Meeting
    2013-02-01 00:00:00    85482        1
    2013-02-01 00:00:00    47228        2
    2013-02-02 00:00:00    12026        4

如果我想获取行,直到列的总和Meeting等于 13。

     DATE                   ETC      Meeting
    2013-02-01 00:00:00    85482        1
    2013-02-01 00:00:00    47228        2
    2013-02-02 00:00:00    12026        4
    2013-02-03 00:00:00    78927        6
4

8 回答 8

9

这是一种无需存储过程的方法:

SET @msum := 0;
SELECT t1.* 
FROM (
    SELECT m.*,  
          (@msum := @msum + m.meetings) AS cumulative_meetings
    FROM meetings m 
    ORDER BY m.date ASC
) t1 
WHERE t1.cumulative_meetings <= 7;
于 2013-02-15T09:14:36.187 回答
8

这是一种适用于 MySQL 的方法:

SELECT
  O.Id,
  O.Type,
  O.MyAmountCol,
  (SELECT
     sum(MyAmountCol) FROM Table1
   WHERE Id <= O.Id) 'RunningTotal'
FROM Table1 O
HAVING RunningTotal <= 7

它涉及计算运行总计并在运行总计小于或等于给定数字时选择记录,在这种情况下7

SQL小提琴

于 2013-02-15T09:01:52.593 回答
5

这一款胜过所有。

SET @runningTotal=0;
SELECT
  O.Id,
  O.Type,
  O.MyAmountCol,
  @runningTotal + O.MyAmountCol as 'RunningTotal',
  @runningTotal := @runningTotal + O.MyAmountCol
FROM Table1 O
HAVING RunningTotal <=7;

SQL小提琴

查看两个查询的执行计划。

于 2016-03-26T10:04:01.243 回答
1

Hemant 你没有说明使用的 RDBMS。这是 t-sql 中的一个脚本,您可以使用它来解决您的问题。

DECLARE @numberToReach INT;
SET @numberToReach = 10; --you can change this

DECLARE @date DATETIME;
DECLARE @etc VARCHAR(20);
DECLARE @meeting INT;
DECLARE @temp_sum INT;

CREATE TABLE #tempTable
    (
        Dates DATETIME,
        Etcs VARCHAR(20),
        Meeting INT,
    )

DECLARE tempcursor CURSOR FOR
        SELECT *
        FROM YourTABLENAME
OPEN tempcursor;
FETCH NEXT FROM tempcursor INTO @date, @etc, @meeting;

WHILE(@@FETCH_STATUS = 0)
BEGIN
    SET @temp_sum = @temp_sum + @meeting;
    IF @temp_sum < @numberToReach 
    BEGIN
        INSERT INTO #tempTable
        (
            Dates,
            Etcs,
            Meeting
        )
        VALUES
        (
            @date, 
            @etc, 
            @meeting
        )

        FETCH NEXT FROM tempcursor INTO @date, @etc, @meeting;
    END 
END

SELECT * FROM #tempTable

CLOSE tempcursor
DEALLOCATE tempcursor

DROP TABLE  #tempTable
于 2013-02-15T08:31:43.350 回答
1

由于没有给出 DBMS,以下是适用于各种 DBMS 的 ANSI SQL 解决方案:

select *
from (
    select date_column, 
           etc, 
           sum(Meeting) over (order by date_column asc) run_sum
    from meetings
) t
where run_sum <= 7
order by date_column asc;

(我使用列名date_column而不是date因为DATE是SQL中的保留字,不应该用作列名以避免出现问题)

于 2013-02-15T10:50:35.480 回答
0

这是一个丑陋的方式:

SELECT *
FROM meetings m1
WHERE (SELECT SUM(m2.Meeting) 
       FROM meetings m2 
       WHERE m2.DATE < m1.DATE OR (m2.DATE = m1.DATE AND m2.ETC >= m1.ETC)) <= 7

排序首先是基于DATE,然后ETC是降序,因为这似乎是我们必须继续进行的。请注意,如果这不是唯一的,您将得到错误的结果

于 2013-02-15T08:14:46.453 回答
0

在Oracle中,我认为你可以使用SUM解析函数。您可以参考这里: http: //www.adp-gmbh.ch/ora/sql/analytical/sum.html

于 2013-02-15T08:51:20.910 回答
-1

如果您使用的是 SQL Server,请使用CROSS APPLY

于 2013-02-15T08:18:03.157 回答