存储过程冗余数据和排序挑战

时间:2013-04-29 21:59:09

标签: sql sql-server pivot

我有一个带有数据字段成为列的数据字段的存储过程。目前返回的数据有一些冗余,并希望看看是否有办法纠正这个问题。存储过程实际上正在提供.rdlc报告,当前输出如下所示:

Period | No Of Interim | Excellent | Very Good | Good | Satisfactory | Unsatisfactory 
-------------------------------------------------------------------------------------
1                                                    1 
12                                            1 
18                   1
18                               1 
18                                                                  1 
19                                                                                  1
19                                                   1 
2                    1 
2                                1 

这就是它需要的样子:

Period | No Of Interim | Excellent | Very Good | Good | Satisfactory | Unsatisfactory 
-------------------------------------------------------------------------------------
1                                                    1 
2                    1           1
12                                            1 
18                   1           1                                  1 
19                                                   1                              1

Period列需要按升序排序,重复实例需要添加到同一行。

负责输出的存储过程和相应视图如下:

BEGIN
DECLARE @cols NVARCHAR(MAX)
DECLARE @query NVARCHAR(MAX)

SET NOCOUNT ON;
SET TRANSACTION ISOLATION LEVEL READ COMMITTED

SELECT @cols = STUFF((SELECT '],[' + [Description] 
    FROM vQualScoringGrade 
    GROUP BY [Description] 
    ORDER BY MAX([orderby]), [Description] 
    FOR XML PATH('')), 1,2,'') + ']'

SET @query = N'SELECT Period, ' + @cols + ' FROM 
(SELECT Period, Description, Value, OrderBy FROM 
vQualScoringGrade) p
PIVOT (SUM([Value]) for [Description] IN ( ' + @cols + ' )) AS pvt ORDER BY Period'

execute(@query)
end

SELECT TOP (100) PERCENT Period, Description, GradeCount AS Value, OrderBy
FROM (SELECT CASE CHARINDEX('.', Number) WHEN 0 THEN Number ELSE 
     REPLACE(LEFT(Number, CHARINDEX('.', Number)), '.', '') 
     END AS Period, 'NoOfInterim' AS Description, COUNT(Number) AS 
        GradeCount, 1 AS OrderBy
FROM vQualScoringExcellent AS vQualScoringExcellent
     GROUP BY Number, Description
UNION
SELECT CASE CHARINDEX('.', Number) WHEN 0 THEN Number ELSE 
     REPLACE(LEFT(Number, CHARINDEX('.', Number)), '.', '') 
     END AS Period, 'Excellent' AS Description, COUNT(Description) AS 
        GradeCount, 2 AS OrderBy
FROM vQualScoringExcellent AS vQualScoringExcellent_1
     GROUP BY Number, Description
UNION
SELECT CASE CHARINDEX('.', Number) WHEN 0 THEN Number ELSE 
     REPLACE(LEFT(Number, CHARINDEX('.', Number)), '.', '') 
     END AS Period, 'VeryGood' AS Description, COUNT(Description) AS 
        GradeCount, 3 AS OrderBy
FROM vQualScoringVeryGood AS vQualScoringVeryGood
     GROUP BY Number, Description
UNION
SELECT CASE CHARINDEX('.', Number) WHEN 0 THEN Number ELSE 
      REPLACE(LEFT(Number, CHARINDEX('.', Number)), '.', '') 
      END AS Period, 'Good' AS Description, COUNT(Description) AS 
         GradeCount, 4 AS OrderBy
FROM vQualScoringGood AS vQualScoringGood
      GROUP BY Number, Description
UNION
SELECT CASE CHARINDEX('.', Number) WHEN 0 THEN Number ELSE 
     REPLACE(LEFT(Number, CHARINDEX('.', Number)), '.', '') 
     END AS Period, 'Satisfactory' AS Description, COUNT(Description) AS 
        GradeCount, 5 AS OrderBy
FROM vQualScoringSatisfactory AS vQualScoringSatisfactory
     GROUP BY Number, Description
UNION
SELECT CASE CHARINDEX('.', Number) WHEN 0 THEN Number ELSE 
     REPLACE(LEFT(Number, CHARINDEX('.', Number)), '.', '') 
     END AS Period, 'Unsatisfactory' AS Description, COUNT(Description) AS 
        GradeCount, 6 AS OrderBy
FROM vQualScoringUnsatisfactory AS vQualScoringUnsatisfactory
     GROUP BY Number, Description) AS QualScoringGrade
ORDER BY OrderBy

1 个答案:

答案 0 :(得分:1)

我猜这个问题是由PIVOT查询中的OrderBy列引起的。这一行是导致问题的原因:

SELECT Period, Description, Value, OrderBy

我会将查询更改为:

SELECT @cols = STUFF((SELECT '],[' + [Description] 
    FROM vQualScoringGrade 
    GROUP BY [Description] 
    ORDER BY MAX([orderby]), [Description] 
    FOR XML PATH('')), 1,2,'') + ']'

SET @query = N'SELECT Period, ' + @cols + ' 
               FROM 
               (
                  SELECT Period, Description, Value
                  FROM vQualScoringGrade
               ) p
              PIVOT 
              (
                 SUM([Value]) 
                 for [Description] IN ( ' + @cols + ' )
              ) AS pvt 
              ORDER BY Period'

execute(@query);

OrderBy列未在外部选择或PIVOT中使用,但您将其包含在子查询中。此列用于PIVOT期间发生的分组。如果值是不同的,那么您将获得多行。

您可以通过在最终选择列表中加入OrderBy来轻松测试。