Sign Up

Sign Up to our social questions and Answers Engine to ask questions, answer people’s questions, and connect with other people.

Have an account? Sign In

Have an account? Sign In Now

Sign In

Login to our social questions & Answers Engine to ask questions answer people’s questions & connect with other people.

Sign Up Here

Forgot Password?

Don't have account, Sign Up Here

Forgot Password

Lost your password? Please enter your email address. You will receive a link and will create a new password via email.

Have an account? Sign In Now

You must login to ask a question.

Forgot Password?

Need An Account, Sign Up Here

Please briefly explain why you feel this question should be reported.

Please briefly explain why you feel this answer should be reported.

Please briefly explain why you feel this user should be reported.

Sign InSign Up

The Archive Base

The Archive Base Logo The Archive Base Logo

The Archive Base Navigation

  • SEARCH
  • Home
  • About Us
  • Blog
  • Contact Us
Search
Ask A Question

Mobile menu

Close
Ask a Question
  • Home
  • Add group
  • Groups page
  • Feed
  • User Profile
  • Communities
  • Questions
    • New Questions
    • Trending Questions
    • Must read Questions
    • Hot Questions
  • Polls
  • Tags
  • Badges
  • Buy Points
  • Users
  • Help
  • Buy Theme
  • SEARCH
Home/ Questions/Q 6567201
In Process

The Archive Base Latest Questions

Editorial Team
  • 0
Editorial Team
Asked: May 25, 20262026-05-25T14:18:29+00:00 2026-05-25T14:18:29+00:00

i have to show running total with the total column in my application …

  • 0

Query Execution Plani have to show running total with the total column in my application … so i have used the following queries for finding the running total… and i find that both are working as per my need . in one i used the left join with group by and in another one i used the sub query .

and now my question is which one is faster when my data grow in thousands daily and if data will be in limit of 1000 or 2000 rows then which one is better … and any other method by which is more faster then these two ????

declare @tmp table(ind int identity(1,1),col1 int)
insert into @tmp
select 2
union
select 4
union
select 7
union 

select 5
union
select 8
union 
select 10



SELECT t1.col1,sum( t2.col1)
FROM @tmp AS t1 LEFT JOIN @tmp t2 ON t1.ind>=t2.ind
group by t1.ind,t1.col1


select t1.col1,(select sum(col1) from  @tmp as t2 where t2.ind<=t1.ind)
from @tmp as t1
  • 1 1 Answer
  • 0 Views
  • 0 Followers
  • 0
Share
  • Facebook
  • Report

Leave an answer
Cancel reply

You must login to add an answer.

Forgot Password?

Need An Account, Sign Up Here

1 Answer

  • Voted
  • Oldest
  • Recent
  • Random
  1. Editorial Team
    Editorial Team
    2026-05-25T14:18:30+00:00Added an answer on May 25, 2026 at 2:18 pm

    A great resource on calculating running totals in SQL Server is this document by Itzik Ben Gan that was submitted to the SQL Server Team as part of his campaign to have the OVER clause extended further from its initial SQL Server 2005 implementation. In it he shows how once you get into tens of thousands of rows cursors out perform set based solutions. SQL Server 2012 did indeed extend the OVER clause making this sort of query much easier.

    SELECT col1,
           SUM(col1) OVER (ORDER BY ind ROWS UNBOUNDED PRECEDING)
    FROM   @tmp 
    

    As you are on SQL Server 2005 however this is not available to you.

    Adam Machanic shows here how the CLR can be used to improve on the performance of standard TSQL cursors.

    For this table definition

    CREATE TABLE RunningTotals
    (
    ind int identity(1,1) primary key,
    col1 int
    )
    

    I create tables with both 2,000 and 10,000 rows in a database with ALLOW_SNAPSHOT_ISOLATION ON and one with this setting off (The reason for this is because my initial results were in a DB with the setting on that led to a puzzling aspect of the results).

    The clustered indexes for all tables just had 1 root page. The number of leaf pages for each is shown below.

    +-------------------------------+-----------+------------+
    |                               | 2,000 row | 10,000 row |
    +-------------------------------+-----------+------------+
    | ALLOW_SNAPSHOT_ISOLATION OFF  |         5 |         22 |
    | ALLOW_SNAPSHOT_ISOLATION ON   |         8 |         39 |
    +-------------------------------+-----------+------------+
    

    I tested the following cases (Links show execution plans)

    1. Left Join and Group By
    2. Correlated subquery 2000 row plan,10000 row plan
    3. CTE from Mikael’s (updated) answer
    4. CTE below

    The reason for inclusion of the additional CTE option was in order to provide a CTE solution that would still work if the ind column was not guaranteed sequential.

    SET STATISTICS IO ON;
    SET STATISTICS TIME ON;
    DECLARE @col1 int, @sumcol1 bigint;
    
    WITH    RecursiveCTE
    AS      (
            SELECT TOP 1 ind, col1, CAST(col1 AS BIGINT) AS Total
            FROM RunningTotals
            ORDER BY ind
            UNION   ALL
            SELECT  R.ind, R.col1, R.Total
            FROM    (
                    SELECT  T.*,
                            T.col1 + Total AS Total,
                            rn = ROW_NUMBER() OVER (ORDER BY T.ind)
                    FROM    RunningTotals T
                    JOIN    RecursiveCTE R
                            ON  R.ind < T.ind
                    ) R
            WHERE   R.rn = 1
            )
    SELECT  @col1 =col1, @sumcol1=Total
    FROM    RecursiveCTE
    OPTION  (MAXRECURSION 0);
    

    All of the queries had a CAST(col1 AS BIGINT) added in order to avoid overflow errors at runtime. Additionally for all of them I assigned the results to variables as above in order to eliminate time spent sending back results from consideration.

    Results

    +------------------+----------+--------+------------+---------------+------------+---------------+-------+---------+
    |                  |          |        |          Base Table        |         Work Table         |     Time        |
    +------------------+----------+--------+------------+---------------+------------+---------------+-------+---------+
    |                  | Snapshot | Rows   | Scan count | logical reads | Scan count | logical reads | cpu   | elapsed |
    | Group By         | On       | 2,000  | 2001       | 12709         |            |               | 1469  | 1250    |
    |                  | On       | 10,000 | 10001      | 216678        |            |               | 30906 | 30963   |
    |                  | Off      | 2,000  | 2001       | 9251          |            |               | 1140  | 1160    |
    |                  | Off      | 10,000 | 10001      | 130089        |            |               | 29906 | 28306   |
    +------------------+----------+--------+------------+---------------+------------+---------------+-------+---------+
    | Sub Query        | On       | 2,000  | 2001       | 12709         |            |               | 844   | 823     |
    |                  | On       | 10,000 | 2          | 82            | 10000      | 165025        | 24672 | 24535   |
    |                  | Off      | 2,000  | 2001       | 9251          |            |               | 766   | 999     |
    |                  | Off      | 10,000 | 2          | 48            | 10000      | 165025        | 25188 | 23880   |
    +------------------+----------+--------+------------+---------------+------------+---------------+-------+---------+
    | CTE No Gaps      | On       | 2,000  | 0          | 4002          | 2          | 12001         | 78    | 101     |
    |                  | On       | 10,000 | 0          | 20002         | 2          | 60001         | 344   | 342     |
    |                  | Off      | 2,000  | 0          | 4002          | 2          | 12001         | 62    | 253     |
    |                  | Off      | 10,000 | 0          | 20002         | 2          | 60001         | 281   | 326     |
    +------------------+----------+--------+------------+---------------+------------+---------------+-------+---------+
    | CTE Alllows Gaps | On       | 2,000  | 2001       | 4009          | 2          | 12001         | 47    | 75      |
    |                  | On       | 10,000 | 10001      | 20040         | 2          | 60001         | 312   | 413     |
    |                  | Off      | 2,000  | 2001       | 4006          | 2          | 12001         | 94    | 90      |
    |                  | Off      | 10,000 | 10001      | 20023         | 2          | 60001         | 313   | 349     |
    +------------------+----------+--------+------------+---------------+------------+---------------+-------+---------+
    

    Both the correlated subquery and the GROUP BY version use “triangular” nested loop joins driven by a clustered index scan on the RunningTotals table (T1) and, for each row returned by that scan, seeking back into the table (T2) self joining on T2.ind<=T1.ind.

    This means that the same rows get processed repeatedly. When the T1.ind=1000 row is processed the self join retrieves and sums all rows with an ind <= 1000, then for the next row where T1.ind=1001 the same 1000 rows are retrieved again and summed along with one additional row and so on.

    The total number of such operations for a 2,000 row table is 2,001,000, for 10k rows 50,005,000 or more generally (n² + n) / 2 which clearly grows exponentially.

    In the 2,000 row case the main difference between the GROUP BY and the subquery versions is that the former has the stream aggregate after the join and so has three columns feeding into it (T1.ind, T2.col1, T2.col1) and a GROUP BY property of T1.ind whereas the latter is calculated as a scalar aggregate, with the stream aggregate before the join, only has T2.col1 feeding into it and has no GROUP BY property set at all. This simpler arrangement can be seen to have a measurable benefit in terms of reduced CPU time.

    For the 10,000 row case there is an additional difference in the sub query plan. It adds an eager spool which copies all the ind,cast(col1 as bigint) values into tempdb. In the case that snapshot isolation is on this works out more compact than the clustered index structure and the net effect is to reduce the number of reads by about 25% (as the base table preserves quite a lot of empty space for versioning info), when this option is off it works out less compact (presumably due to the bigint vs int difference) and more reads result. This reduces the gap between the sub query and group by versions but the sub query still wins.

    The clear winner however was the Recursive CTE. For the “no gaps” version logical reads from the base table are now 2 x (n + 1) reflecting the n index seeks into the 2 level index to retrieve all of the rows plus the additional one at the end that returns nothing and terminates the recursion. That still meant 20,002 reads to process a 22 page table however!

    Logical work table reads for the recursive CTE version are very high. It seems to work out at 6 worktable reads per source row. These come from the index spool that stores the output of the previous row then is read from again in the next iteration (good explanation of this by Umachandar Jayachandran here). Despite the high number this is still the best performer.

    • 0
    • Reply
    • Share
      Share
      • Share on Facebook
      • Share on Twitter
      • Share on LinkedIn
      • Share on WhatsApp
      • Report

Sidebar

Related Questions

I have the following JavaScript for show running line: <script type=text/javascript language=javascript> //Change script's
I have some text in a UITextView, that I would like to have show
I have Divs show and hide (with an animation) using the following script (I
I have to show records in such a way that some selected records should
I have a show/hide toggle working well in multiple instances (thanks to help here
I have an application in which I have to show the alerts for user
I have created an application that looks like a news ticker. It suppose to
I have a multicolumn crystal reports, Now i want to display running total for
We have a swing based application that does complex processing on data. One of
In an application I'm developing, I have a main form that simply sits there

Explore

  • Home
  • Add group
  • Groups page
  • Communities
  • Questions
    • New Questions
    • Trending Questions
    • Must read Questions
    • Hot Questions
  • Polls
  • Tags
  • Badges
  • Users
  • Help
  • SEARCH

Footer

© 2021 The Archive Base. All Rights Reserved
With Love by The Archive Base

Insert/edit link

Enter the destination URL

Or link to existing content

    No search term specified. Showing recent items. Search or use up and down arrow keys to select an item.