SQL Query: Pivot with Grand Total Column and Row
Introduction
Microsoft SQL Server has introduced thePIVOT
and UNPIVOT
commands as enhancements to T-SQL with the release of Microsoft SQL Server 2005. Pivot Table in SQL has the ability to display data in custom aggregation.PIVOT
rotates a table-valued expression by turning the unique values from one column in the expression into multiple columns in the output, and performs aggregations where they are required on any remaining column values that are wanted in the final output. UNPIVOT
performs the opposite operation to PIVOT
by rotating columns of a table-valued expression into column values.More information on Pivot.
In this article, I’m concentrating on how to display Grand Total row and column for the pivot table as shown in the below grid view:
Here in the grid, I’m showing the number of matches played by a team in each month. At last, I need to show the grand total for each team (Last column) - this total gives the year count for the teams. In the same way, I need a grand total row as last row in grid, which will give the number of matches played by all the teams for that particular month.
Background
Usually as a .NET developer, first I will get the pivot table from the stored procedure to dataset and grand total row and column. I will manipulate in the C#/ VB code usingDatatable
(Datatable compute method for better performance). But in this article, I want to show how to get the grand total row and column from the stored procedure, so that we can directly display data in the grid without any manipulation in the C#/VB.How It Works
Here I’m using pivot with dynamic columns, most questions were about the column list in thePIVOT
statement. This list is fixed, but many times the new columns are determined by the report at a later stage. This problem is easily solved when we mix pivots with dynamic SQL, so here is a very simple script about how to dynamically generate the pivot
statement:DECLARE @cols NVARCHAR (MAX)
SELECT @cols = COALESCE (@cols + ',[' + colName + ']', '[' + colName + ']')
FROM Table1
WHERE Conditions
ORDER BY colName
PRINT @cols
The above script gives you the list of columns in string
format with commas. Example: [Jan], [Feb], [Mar], [Apr], [May], [Jun], [Jul], [Aug], [Sept], [Oct], [Nov], [Dec]
. If you pass this dynamic query to your pivot, then your pivot columns will display dynamically.The below SQL script creates the stored procedure which returns pivot table as output.
CREATE PROCEDURE pivot_TeamVsMatches
AS
/*First get the dynamic columns query*/
DECLARE @columnHeaders NVARCHAR (MAX)
SELECT @columnHeaders = _
COALESCE (@columnHeaders + ',[' + month + ']', '[' + month + ']')
FROM tbl_Matches
ORDER BY month
DECLARE @FinalQuery NVARCHAR(MAX)
SET @FinalQuery = ‘SELECT *
FROM
(SELECT Team,
Month
FROM tbl_Matches
) A
PIVOT
(
COUNT(*)
FOR ColName
IN (‘+@columnHeaders +’)
) B
ORDER BY Team’
PRINT ‘Pivot Queuery :’+ @FinalQuery
EXECUTE (@FinalQuery)
GO
This stored procedure returns pivot table as below:Now to get the grand total and row, we need to form the
COALESCE
query. As shown below…/* GRAND TOTAL COLUMN */
DECLARE @GrandTotalCol NVARCHAR (MAX)
SELECT @GrandTotalCol = COALESCE (@GrandTotalCol + ‘ISNULL _
([' + CAST (Month AS VARCHAR) +'],0) + ', 'ISNULL([' + CAST(Month AS VARCHAR)+ '],0) + ')
FROM tbl_Matches ORDER BY Month
SET @GrandTotalCol = LEFT (@GrandTotalCol, LEN (@GrandTotalCol)-1)
The above query returns as below…@GrandTotalCol = ISNULL ([' + CAST (Jan AS VARCHAR) +'],0) + ISNULL _
([' + CAST (Feb AS VARCHAR) +'],0) + ISNULL ([' + CAST (March AS VARCHAR) +'],0) + _
…………. + ISNULL ([' + CAST (Dec AS VARCHAR) +'],0).
/* GRAND TOTAL ROW */
DECLARE @GrandTotalRow NVARCHAR(MAX)
SELECT @GrandTotalRow = COALESCE(@GrandTotalRow + ',ISNULL(SUM([' + _
CAST(Month AS VARCHAR)+']),0)', 'ISNULL(SUM([' + CAST(Month AS VARCHAR)+']),0)')
FROM tbl_Matches ORDER BY Month
The above query returns as below…@GrandTotalRow = ISNULL(SUM([' + CAST(Jan AS VARCHAR)+']),0) + _
ISNULL(SUM([' + CAST(Feb AS VARCHAR)+']),0) + ……… + _
ISNULL(SUM([' + CAST(Dec AS VARCHAR)+']),0).
The above COALESCE string
s need to be used in dynamic pivot query…Below is the stored procedure which will give the total output of our requirement.
CREATE PROCEDURE pivot_TeamVsMatches
AS
/* COLUMNS HEADERS */
DECLARE @columnHeaders NVARCHAR (MAX)
SELECT @columnHeaders = COALESCE (@columnHeaders _
+ ',[' + month + ']', '[' + month + ']')
FROM tbl_Matches
ORDER BY month
/* GRAND TOTAL COLUMN */
DECLARE @GrandTotalCol NVARCHAR (MAX)
SELECT @GrandTotalCol = COALESCE (@GrandTotalCol + ‘ISNULL ([' + _
CAST (Month AS VARCHAR) +'],0) + ', 'ISNULL([' + CAST(Month AS VARCHAR)+ '],0) + ')
FROM tbl_Matches ORDER BY Month
SET @GrandTotalCol = LEFT (@GrandTotalCol, LEN (@GrandTotalCol)-1)
/* GRAND TOTAL ROW */
DECLARE @GrandTotalRow NVARCHAR(MAX)
SELECT @GrandTotalRow = COALESCE(@GrandTotalRow + ',ISNULL(SUM([' + _
CAST(Month AS VARCHAR)+']),0)', 'ISNULL(SUM([' + CAST(Month AS VARCHAR)+']),0)')
FROM tbl_Matches ORDER BY Month
/* MAIN QUERY */
DECLARE @FinalQuery NVARCHAR (MAX)
SET @FinalQuery = ‘SELECT *, ('+ @GrandTotalCol + ') _
AS [Grand Total] INTO #temp_MatchesTotal
FROM
(SELECT Team,
Month
FROM tbl_Matches
) A
PIVOT
(
COUNT (*)
FOR ColName
IN (‘+@columnHeaders +’)
) B
ORDER BY Team
SELECT * FROM #temp_MatchesTotal UNION ALL
SELECT ''Grand Total'','''','+@GrandTotalRow +', _
ISNULL (SUM([Grand Total]),0) FROM #temp_MatchesTotal
DROP TABLE #temp_MatchesTotal'
-- PRINT 'Pivot Query '+@FinalQuery
EXECUTE(@PivotQuery)
GO
Result as below…Here in this stored procedure, I have used temporary table to get the grand total row. I did the
UNION ALL
with temporary table; don’t forget to drop the temporary table.For any queries & suggestions, mail me @ narapareddy.shyam@gmail.com. By using some third party controls also we can achieve this, but here I concentrated only on typical/ normal SQL query. If anybody has any efficient and different ways, kindly share with me.
Thanks again. Bye.
Once again thanks a lot to Mr.shyamprasad for this wonderful article.It helped me lot. I have seen lot of peoples asked for db script and details .Here is the sample I have done with help of this article. DB SCRIPT ------------------ CREATE TABLE [dbo].[TestPivot](
[Team] [nvarchar](50) NULL,
[month] [nvarchar](50) NULL,
[rank] [int] NULL
) ON [PRIMARY]
GO
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'INDIA', N'JAN', 10)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'INDIA', N'FEB', 20)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'INDIA', N'MAR', 30)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'INDIA', N'APR', 20)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'INDIA', N'MAY', 10)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'INDIA', N'JUN', 20)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'JAPAN', N'JAN', 10)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'JAPAN', N'MAY', 52)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'JAPAN', N'JUN', 52)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'JAPAN', N'APR', 10)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'AMERICA', N'JAN', 50)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'AMERICA', N'FEB', 20)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'AMERICA', N'MAR', 12)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'AMERICA', N'MAY', 5)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'AMERICA', N'JUN', 34)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'JERMANY', N'JUN', 23)
INSERT [dbo].[TestPivot] ([Team], [month], [rank]) VALUES (N'JERMANY', N'JAN', 10)
And SP ------------ ALTER PROCEDURE pivot_TeamVsMatches3
AS
/* COLUMN HEADERS*/
DECLARE @columnHeaders NVARCHAR (MAX)
SELECT @columnHeaders = COALESCE ( (@columnHeaders) + ',[' + month + ']', '[' + month + ']')
FROM (Select Distinct month from TestPivot) as PIVT
Print @columnHeaders
/* GRAND TOTAL COLUMN */
DECLARE @GrandTotalCol NVARCHAR (MAX)
SELECT @GrandTotalCol = COALESCE (@GrandTotalCol + 'ISNULL ([' + CAST (month AS VARCHAR) +'],0) + ', 'ISNULL([' + CAST(month AS VARCHAR)+ '],0) + ')
FROM (Select Distinct month from TestPivot) as PIVT ORDER BY month
SET @GrandTotalCol = LEFT (@GrandTotalCol, LEN (@GrandTotalCol)-1)
/* GRAND TOTAL ROW */
DECLARE @GrandTotalRow NVARCHAR(MAX)
SELECT @GrandTotalRow = COALESCE(@GrandTotalRow + ',ISNULL(SUM([' + CAST(month AS VARCHAR)+']),0)', 'ISNULL(SUM([' + CAST(month AS VARCHAR)+']),0)')
FROM (Select Distinct month from TestPivot) as PIVT ORDER BY month
/* MAIN QUERY */
DECLARE @FinalQuery NVARCHAR (MAX)
SET @FinalQuery = 'SELECT *, ('+ @GrandTotalCol + ') AS [GRANT TOTAL] INTO #temp_MatchesTotal
FROM
(SELECT Team,month,rank
FROM TestPivot) A
PIVOT
(
SUM(rank)
FOR month
IN (' +@columnHeaders +')
) B
ORDER BY Team
SELECT * FROM #temp_MatchesTotal UNION ALL SELECT ''GRANT TOTAL'', '+@GrandTotalRow +',ISNULL (SUM([GRANT TOTAL]),0) FROM #temp_MatchesTotal
DROP TABLE #temp_MatchesTotal'
PRINT 'Pivot Query '+@FinalQuery
EXECUTE(@FinalQuery)
GO
EXEC pivot_TeamVsMatches3
|
0 Comments:
Post a Comment
Subscribe to Post Comments [Atom]
<< Home