如何在数据透视表上将多行折叠为一列

时间:2019-10-03 16:26:48

标签: sql duplicates pivot unpivot

我对SQL代码没有经验,所以去吧。

我想从表中提取数据,并使用PIVOT将值转换为列标题。

我已经设法使用PIVOT命令(和动态字段名称)实现了我想要的功能,但我希望针对单个引用的所有值都将显示在一行上。但是,对于原始表中的每个代码,我都会获得一行代码(请参见下面的结果)。

使用CROSS APPLY似乎是一种方法(在我的代码示例中已注释掉),但是我遇到了一些问题:

  1. 我无法理解它是如何工作的,因此根据我的情况调整在线示例涉及到一些猜测工作。
  2. 我根据此站点上的另一篇文章在命令中输入了(我认为是)正确的引用: Pivot a table on a value but group the data on one line by another? 他们声称VALUES中的两个值都必须是相同的数据类型,才能正常运行流程。但是,将'value'设置为varchar(10)会使SUM函数无法在PIVOT语句中运行。以十进制/浮点数运行将产生所有NULL结果。
  3. 将“值”设置为十进制,包括GROUP BY语句,确实会产生一行数据,但是(如第2点所示)所有代码值均为NULL。

希望所有这些都是有道理的。请建议我如何将以下结果折叠为一行。预先感谢。

DECLARE @CODES nvarchar(max)
select @CODES =
    stuff(
    (
    select distinct ',[' + code + ']'
    from codetable
    for xml path('')
    ),
    1,1,'')

DECLARE @SQL nvarchar(max)
SET @SQL = 
N'
SELECT
    ref,
    ' + @CODES + ',
FROM
    codetable
    --CROSS APPLY
    --(
 --   VALUES
 --     (''code'', convert(varchar(10), code, 120)),
 --     (''value'', convert(decimal, value))
    --) CA (CODE, VAL)
PIVOT(SUM(value)
    FOR code IN (' + @CODES + '))
    AS PVTTable2
WHERE
    ref = ''101'' AND end_date IS NULL
GROUP BY ref,' + @CODES + '
'
exec sp_executesql @SQL
My original table is like this:
-------------------------
|ref    |CODES  |VALUE  |
-------------------------
|101    |CODE4  |20     |
|101    |CODE1  |2      |
|101    |CODE7  |38     |


The results I get are:
-------------------------------------------------------------------------
|ref    |CODE1  |CODE2  |CODE3  |CODE4  |CODE5  |CODE6  |CODE7  |CODE8  |
-------------------------------------------------------------------------
|101    |NULL   |NULL   |NULL   |20     |NULL   |NULL   |NULL   |NULL   |
|101    |2      |NULL   |NULL   |NULL   |NULL   |NULL   |NULL   |NULL   |
|101    |NULL   |NULL   |NULL   |NULL   |NULL   |NULL   |38     |NULL   |

2 个答案:

答案 0 :(得分:0)

GROUP BY ref

DECLARE @CODES nvarchar(max)
select @CODES =
    stuff(
    (
    select distinct ',[' + code + ']'
    from codetable
    for xml path('')
    ),
    1,1,'');

DECLARE @CODESGrp nvarchar(max)
select @CODESGrp =
    stuff(
    (
    select distinct ',min([' + code + '])'
    from codetable
    for xml path('')
    ),
    1,1,''); 

DECLARE @SQL nvarchar(max);
SET @SQL = 'SELECT  ref,' + @CODESGrp 
+ ' FROM  codetable '
+ ' PIVOT(SUM(value) FOR code IN (' + @CODES + ')) AS PVTTable2'
+ ' WHERE ref = 101 GROUP BY ref' ;

exec sp_executesql @SQL;

答案 1 :(得分:0)

数据透视应写为:

DECLARE @cols AS NVARCHAR(MAX),@query  AS NVARCHAR(MAX);

SET @cols = STUFF((SELECT distinct ',' + QUOTENAME(T.CODES) 
            FROM codetable T
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'')


set @query = 'SELECT ref, ' + @cols + ' from 
            (
             SELECT
                 ref, --< grouping column >,
                 CODES, --< spreading column >, 
                 VALUE --< aggregation column >
             FROM codetable
           ) PivotData
            pivot 
            (
                sum(VALUE)
                for CODES in (' + @cols + ')
            ) p '

exec sp_executesql @query

sample code here..