Как заменить NULL на 0 в динамическом стержне и рассчитать накопительный в SQL 2012 - PullRequest
0 голосов
/ 20 июня 2019

Вывод моего динамического пивота выглядит следующим образом:

enter image description here

Я предполагаю, что вместо NULL я получу 0.Тогда EffectiveMMYY будет сгруппировано и не будет иметь дубликатов.

Итак, как я могу заменить NULL на 0?

Без совокупного итогового вывода должно выглядеть так:

enter image description here

При совокупном общем выводе должно выглядеть так:

enter image description here

ссылка на dbfidle

https://dbfiddle.uk/?rdbms=sqlserver_2016&fiddle=b2e9d0a6fc58705b8f310babbe3f158e

Это пример запроса:

IF OBJECT_ID('tempdb..##DynPivot') IS NOT NULL DROP TABLE ##DynPivot
DECLARE @Table1 TABLE (ID int, PolicyNumber varchar(50),EffectiveDate date, LossDate date)
INSERT INTO @Table1
(
    ID,
    PolicyNumber,
    EffectiveDate,
    LossDate
)
VALUES
            ( 1, 'Pol1', '2017-02-01',  '2017-02-05'),  
            ( 2, 'Pol2', '2017-03-04',   '2017-03-18'), 
            ( 3, 'Pol3', '2017-03-11', '2017-03-22'),
            ( 4, 'Pol4', '2017-03-11',  NULL),  
            ( 5, 'Pol5', '2017-03-15', '2017-04-22'),
            ( 6, 'Pol6', '2017-04-18', '2017-04-19'),   
            ( 7, 'Pol7', '2017-04-22', '2017-04-29'),
            ( 8, 'Pol8', '2017-04-15', '2017-04-18'),   
            ( 9, 'Pol9', '2017-12-15', '2018-01-23'),
            ( 10, 'Pol10', '2017-02-13', '2017-03-11'),
            ( 11, 'Pol11', '2017-02-18', '2017-03-22'),
            ( 12, 'Pol12', '2017-02-17', '2018-01-05'),
            ( 13, 'Pol13', '2017-05-28', '2017-05-30')

SELECT  
        FORMAT(EffectiveDate, 'MMM') + '-'+ FORMAT(EffectiveDate, 'yy') AS  'EffectiveMMYY',
        YEAR(EffectiveDate) AS EffYear, MONTH(EffectiveDate) AS EffMonth,
        YEAR(LossDate) AS LossYear, MONTH(LossDate) AS LossMonth,

        FORMAT(LossDate, 'MMM') + '-'+ FORMAT(LossDate, 'yy') AS  'LossMMYY',
        count(PolicyNumber) AS PolicyCout
into ##DynPivot
FROM @Table1 

WHERE LossDate IS NOT NULL
GROUP BY FORMAT(EffectiveDate, 'MMM') + '-'+ FORMAT(EffectiveDate, 'yy'),
         FORMAT(LossDate, 'MMM') + '-'+ FORMAT(LossDate, 'yy'),
         YEAR(EffectiveDate),
         MONTH(EffectiveDate),
         YEAR(LossDate), MONTH(LossDate)
ORDER BY FORMAT(EffectiveDate, 'MMM') + '-'+ FORMAT(EffectiveDate, 'yy')

Declare @SQL varchar(max) = Stuff((Select  ',' + QuoteName(LossMMYY) From ##DynPivot group by LossMMYY Order by min(LossYear), min(LossMonth) For XML Path('')),1,1,'') 

select @SQL = '
select  EffectiveMMYY,' + @SQL + '
from ##DynPivot
pivot ( sum(PolicyCout) for LossMMYY IN (' + @SQL + ') ) p
order by EffYear,EffMonth
'Exec(@SQL);

Ответы [ 3 ]

2 голосов
/ 20 июня 2019

Как я уже говорил, если вы используете кросс-таб (который я лично считаю намного лучше, чем оператор PIVOT по нескольким причинам), такой проблемы не существует:

CREATE TABLE Table1 (ID int, PolicyNumber varchar(50),EffectiveDate date, LossDate date);
INSERT INTO Table1
(
    ID,
    PolicyNumber,
    EffectiveDate,
    LossDate
)
VALUES
            ( 1, 'Pol1', '2017-02-01',  '2017-02-05'),  
            ( 2, 'Pol2', '2017-03-04',   '2017-03-18'), 
            ( 3, 'Pol3', '2017-03-11', '2017-03-22'),
            ( 4, 'Pol4', '2017-03-11',  NULL),  
            ( 5, 'Pol5', '2017-03-15', '2017-04-22'),
            ( 6, 'Pol6', '2017-04-18', '2017-04-19'),   
            ( 7, 'Pol7', '2017-04-22', '2017-04-29'),
            ( 8, 'Pol8', '2017-04-15', '2017-04-18'),   
            ( 9, 'Pol9', '2017-12-15', '2018-01-23'),
            ( 10, 'Pol10', '2017-02-13', '2017-03-11'),
            ( 11, 'Pol11', '2017-02-18', '2017-03-22'),
            ( 12, 'Pol12', '2017-02-17', '2018-01-05'),
            ( 13, 'Pol13', '2017-05-28', '2017-05-30');
DECLARE @SQL nvarchar(MAX);

SET @SQL = N'SELECT ISNULL(STUFF(CONVERT(varchar(9),DATEFROMPARTS(YEAR(EffectiveDate), MONTH(EffectiveDate), 1),6),1,3,''''),''Total'') AS EffectiveMonth,' + NCHAR(13) + NCHAR(10) +
           STUFF((SELECT N',' + NCHAR(13) + NCHAR(10) + 
                         N'       COUNT(CASE WHEN DATEFROMPARTS(YEAR(LossDate), MONTH(LossDate), 1) = ' + QUOTENAME(CONVERT(varchar(8),V.LossDate,112),'''') + N'THEN 1 END) AS ' + QUOTENAME(STUFF(CONVERT(varchar(9),V.LossDate,6),1,3,''))
                  FROM Table1 T1
                       CROSS APPLY(VALUES(DATEFROMPARTS(YEAR(LossDate), MONTH(LossDate), 1)))V(LossDate)
                  GROUP BY V.LossDate
                  ORDER BY V.LossDate
                  FOR XML PATH(N''),TYPE).value('.','nvarchar(MAX)'),1,3,N'') + N',' + NCHAR(13) + NCHAR(10) +                  
           N'       COUNT(*) AS Total' + NCHAR(13) + NCHAR(10) +
           N'FROM Table1'  + NCHAR(13) + NCHAR(10) +
           N'GROUP BY DATEFROMPARTS(YEAR(EffectiveDate), MONTH(EffectiveDate), 1) WITH ROLLUP;';

PRINT @SQL;

EXEC sp_executesql @SQL;
GO
DROP TABLE Table1;

DB <> скрипка

1 голос
/ 20 июня 2019

Я собирался отредактировать ответ Ларну, который очень похож на мой, но не будет включать в себя совокупный счет.Он также использует кросс-таблицы, которые обеспечивают большую гибкость, чего нет у оператора PIVOT.

DECLARE @SQL1 NVARCHAR(MAX);

SELECT @SQL1 = 'SELECT STUFF( CONVERT( varchar(11) , EffectiveDate, 100), 4, 4, ''-'') AS EffectiveMMYY, ' 
            + ( SELECT DISTINCT CHAR(10) + 
                '    COUNT( CASE WHEN LossDate < ''' + CONVERT( char(8) , DATEADD(MM, DATEDIFF( MM, -1, LossDate), 0), 112) + ''' THEN PolicyNumber END ) AS ' + QUOTENAME(STUFF( CONVERT( varchar(11) , LossDate, 100), 4, 4, '-')) + ','
                FROM Table1
                WHERE LossDate IS NOT NULL
                ORDER BY 1
                FOR XML PATH(''), TYPE).value('./text()[1]', 'varchar(max)')
            +'
    COUNT(LossDate) AS Total
FROM Table1
GROUP BY STUFF( CONVERT( varchar(11) , EffectiveDate, 100), 4, 4, ''-''), CONVERT( char(6) , EffectiveDate, 112)
ORDER BY CONVERT( char(6) , EffectiveDate, 112)';

EXEC sp_executesql @SQL1;
1 голос
/ 20 июня 2019

Вы перевернули всю таблицу, поэтому у вас есть две строки для Mar-17 и т. Д. Я объявил дополнительную переменную, которая добавляет isnull к каждому столбцу.

IF OBJECT_ID('tempdb..##DynPivot') IS NOT NULL DROP TABLE ##DynPivot
DECLARE @Table1 TABLE (ID int, PolicyNumber varchar(50),EffectiveDate date, LossDate date)
INSERT INTO @Table1
(
    ID,
    PolicyNumber,
    EffectiveDate,
    LossDate
)
VALUES
            ( 1, 'Pol1', '2017-02-01',  '2017-02-05'),  
            ( 2, 'Pol2', '2017-03-04',   '2017-03-18'), 
            ( 3, 'Pol3', '2017-03-11', '2017-03-22'),
            ( 4, 'Pol4', '2017-03-11',  NULL),  
            ( 5, 'Pol5', '2017-03-15', '2017-04-22'),
            ( 6, 'Pol6', '2017-04-18', '2017-04-19'),   
            ( 7, 'Pol7', '2017-04-22', '2017-04-29'),
            ( 8, 'Pol8', '2017-04-15', '2017-04-18'),   
            ( 9, 'Pol9', '2017-12-15', '2018-01-23'),
            ( 10, 'Pol10', '2017-02-13', '2017-03-11'),
            ( 11, 'Pol11', '2017-02-18', '2017-03-22'),
            ( 12, 'Pol12', '2017-02-17', '2018-01-05'),
            ( 13, 'Pol13', '2017-05-28', '2017-05-30')

SELECT  
        FORMAT(EffectiveDate, 'MMM') + '-'+ FORMAT(EffectiveDate, 'yy') AS  'EffectiveMMYY',
        YEAR(EffectiveDate) AS EffYear, MONTH(EffectiveDate) AS EffMonth,
        YEAR(LossDate) AS LossYear, MONTH(LossDate) AS LossMonth,

        FORMAT(LossDate, 'MMM') + '-'+ FORMAT(LossDate, 'yy') AS  'LossMMYY',
        count(PolicyNumber) AS PolicyCout
into ##DynPivot
FROM @Table1 

WHERE LossDate IS NOT NULL
GROUP BY FORMAT(EffectiveDate, 'MMM') + '-'+ FORMAT(EffectiveDate, 'yy'),
         FORMAT(LossDate, 'MMM') + '-'+ FORMAT(LossDate, 'yy'),
         YEAR(EffectiveDate),
         MONTH(EffectiveDate),
         YEAR(LossDate), MONTH(LossDate)
ORDER BY FORMAT(EffectiveDate, 'MMM') + '-'+ FORMAT(EffectiveDate, 'yy')

Declare @SQL varchar(max) = Stuff((Select  ',' + QuoteName(LossMMYY) From ##DynPivot group by LossMMYY Order by min(LossYear), min(LossMonth) For XML Path('')),1,1,'') 
Declare @SQL0 varchar(max) = Stuff((Select  ',' + 'isnull(' + QuoteName(LossMMYY) + ',0) as ' + QuoteName(LossMMYY) From ##DynPivot group by LossMMYY Order by min(LossYear), min(LossMonth) For XML Path('')),1,1,'') 

select @SQL
select @SQL = '
select  EffectiveMMYY,' + @SQL0 + '
from (select EffectiveMMYY, LossMMYY, PolicyCout from ##DynPivot) d
pivot ( sum(PolicyCout) for LossMMYY IN (' + @SQL + ') ) p
order by EffectiveMMYY
'exec(@SQL);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...