【问题标题】:SQL running sum for an MVC applicationMVC 应用程序的 SQL 运行总和
【发布时间】:2012-12-08 17:08:18
【问题描述】:

我需要一种更快的方法来计算和显示运行总和。

这是一个 MVC telerik 网格,它查询一个使用子查询生成运行总和的视图。查询需要 73 秒才能完成,这是不可接受的。 (每次用户点击“刷新预测表”,重新填充网格需要 73 秒。)

查询如下所示:

SELECT outside.EffectiveDate
[omitted for clarity]
       ,(
              SELECT SUM(b.Amount)
              FROM vCI_UNIONALL inside
              WHERE inside.EffectiveDate <= outside.EffectiveDate
              ) AS RunningBalance
[omitted for clarity]
FROM vCI_UNIONALL outside

某些项目上的“EffectiveDate”可以一直更改......可以添加新项目等。我当然需要一些可以即时计算运行总和的东西(当点击刷新按钮时)。存储过程或其他视图...?请指教。

解决方案:(众多之一,这个比子查询快几个数量级)

创建一个包含视图中所有列的新表除了RunningTotal col。创建一个存储过程,首先截断表,然后使用 SELECT 的所有列截断表 INSERT INTO,不使用运行总和列。

使用更新局部变量方法:

DECLARE @Amount DECIMAL(18,4)
SET @Amount = 0
UPDATE TABLE_YOU_JUST_CREATED SET RunningTotal = @Amount, @Amount = @Amount + ISNULL(Amount,0)

创建一个每天运行一次存储过程的任务代理。为您的所有报告使用TABLE_YOU_JUST_CREATED

【问题讨论】:

    标签: sql-server asp.net-mvc linq sql-server-2008 tsql


    【解决方案1】:

    看看这个帖子 Calculate a Running Total in SQL Server

    如果您有 SQL Server Denali,则可以使用新的窗口函数。

    在 SQL Server 2008 R2 中,我建议您使用递归公用表表达式。 CTE 中的小问题是,对于快速查询,您必须有没有间隙的标识列(1、2、3,...),如果您没有这样的列,则必须创建一个临时或变量表,其中包含列并将您的数据移到那里。

    CTE 方法将是这样的

    declare @Temp_Numbers (RowNum int, Amount <your type>, EffectiveDate datetime)
    
    insert into @Temp_Numbers (RowNum, Amount, EffectiveDate)
    select row_number() over (order by EffectiveDate), Amount, EffectiveDate
    from vCI_UNIONALL
    
    -- you can also use identity
    -- declare @Temp_Numbers (RowNum int identity(1, 1), Amount <your type>, EffectiveDate datetime)
    -- insert into @Temp_Numbers (Amount, EffectiveDate)
    -- select Amount, EffectiveDate
    -- from vCI_UNIONALL
    -- order by EffectiveDate
    
    ;with 
    CTE_RunningTotal
    as
    (
        select T.RowNum, T.EffectiveDate, T.Amount as Total_Amount
        from @Temp_Numbers as T
        where T.RowNum = 1
        union all
        select T.RowNum, T.EffectiveDate, T.Amount + C.Total_Amount as Total_Amount
        from CTE_RunningTotal as C
            inner join @Temp_Numbers as T on T.RowNum = C.RowNum + 1
    )
    select C.RowNum, C.EffectiveDate, C.Total_Amount
    from CTE_RunningTotal as C
    option (maxrecursion 0)
    

    对于重复的 EffectiveDate 值可能存在一些问题,这取决于您希望如何使用它们 - 您是希望它们任意排序还是希望它们具有相等的数量?

    【讨论】:

    • 谢谢,我会看看那个帖子。我想我不能干净地使用 CTE,因为这是我正在查询的视图。该视图查询 4,5 个表以获取项目,因此它们的共同标识列是“EffectiveDate”列,这肯定会有差距。
    • 您使用的是 Denali 还是 2008(或更早版本)?
    • EffectiveDate 列有索引吗?
    • 我没有。如何为它创建索引?这就是我的视图的样子:pastebin.com/qRuJm53q
    • 您的问题可能在 QB_INVOICES_HEADER 部分的计算 EffectiveDate 列中 - 找到它的最简单方法是在 union 中评论此部分并尝试运行查询以查看它的执行速度。跨度>
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2020-02-28
    • 2012-09-25
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2023-03-16
    • 1970-01-01
    相关资源
    最近更新 更多