2 votes

Combiner plusieurs lignes avec COALESCE

Utilisation de SQL-Server 2012

J'ai le tableau suivant :

Id      Description 
6192    Salzburg
6193    Salzburg
6194    Salzburg
6196    Innsbruck
6197    Innsbruck
6198    Innsbruck
6199    Innsbruck
6201    Bregenz
6202    Bregenz
6203    Bregenz

Je veux sélectionner chaque "Description" distincte avec tous les identifiants réunis en une seule chaîne :

    Description     Ids
    Salzburg        '6192,6193,6194'
    Innsbruck       '6196,6197,6198'

J'ai vu un code similaire sur ce site [ Comment concaténer le texte de plusieurs lignes en une seule chaîne de texte dans le serveur SQL ? mais je n'ai pas encore réussi à trouver une solution à mon problème (je ne veux pas utiliser XML Path !). Voici ce que j'ai essayé jusqu'à présent :

DECLARE @ids AS Nvarchar(MAX)
SELECT  @ids = COALESCE(@ids + ',', '') + CAST(t.Id AS nvarchar(5))
    FROM (SELECT tmp.Id FROM (SELECT id, [Description] FROM tblMasterPropValues WHERE IdCategory = 253 AND IsActive = 1) as tmp
        WHERE [Description] = tmp.[Description]) AS t
SELECT @ids
--SELECT DISTINCT [Description], @ids AS IDs FROM tblMasterPropValues WHERE IdCategory = 253 AND IsActive = 1 AND Id IN (@ids)

Je n'arrive pas à m'y retrouver, et j'apprécierais toute aide à ce sujet.

1voto

fa06 Points 13375

Vous pouvez essayer d'utiliser STUFF() fonction

SELECT description,  Ids = STUFF(
             (SELECT ',' + Id
              FROM tblMasterPropValues t1
              WHERE t1.description = t2.description
              FOR XML PATH (''))
             , 1, 1, '') from tblMasterPropValues t2
group by description;

1voto

Yogesh Sharma Points 29348

Pour cela FOR XML PATH() est la bonne clause, donc, vous pouvez faire :

SELECT DISTINCT v.description, STUFF(v1.ids, 1, 1, '''') + '''' 
FROM tblMasterPropValues v CROSS APPLY
     (SELECT ', '+ CAST(v1.Id AS VARCHAR(255))
      FROM tblMasterPropValues v1
      WHERE v1.description = v.description 
      FOR XML PATH('')
     ) v1(ids);

1voto

Serkan Arslan Points 8342

Vous pouvez également le faire en utilisant un CTE récursif.

DECLARE @tblMasterPropValues TABLE (Id INT, Description VARCHAR(20))
INSERT INTO @tblMasterPropValues VALUES
(6192 , 'Salzburg'),
(6193 , 'Salzburg'),
(6194 , 'Salzburg'),
(6196 , 'Innsbruck'),
(6197 , 'Innsbruck'),
(6198 , 'Innsbruck'),
(6199 , 'Innsbruck'),
(6201 , 'Bregenz'),
(6202 , 'Bregenz'),
(6203 , 'Bregenz')

;WITH Tbl AS 
(
    SELECT 
        *, 
        ROW_NUMBER() OVER(PARTITION BY Description ORDER BY Id) AS RN,
        COUNT(*) OVER(PARTITION BY Description) AS CNT
    FROM @tblMasterPropValues 
)
, Rcr AS (
    SELECT *, CAST(Id AS varchar(max)) Ids 
    FROM Tbl WHERE RN = 1
        UNION ALL
    SELECT T.*, Rcr.Ids + ',' + CAST(T.Id AS VARCHAR(10)) Ids 
    FROM Rcr 
        INNER JOIN Tbl T ON T.RN = Rcr.RN + 1 and Rcr.Description = T.Description
)
SELECT RN, Description, Ids FROM Rcr 
WHERE RN = CNT

Résultat :

Description          Ids
-------------------- -----------------------
Salzburg             6192,6193,6194
Innsbruck            6196,6197,6198,6199
Bregenz              6201,6202,6203

0voto

DineshDB Points 3881

Essayez ça :

DECLARE @Table TABLE(ID INT, Description VARCHAR(25))
INSERT INTO @Table
VALUES (6192,'Salzburg' )
,(6193,'Salzburg' )
,(6194,'Salzburg' )
,(6196,'Innsbruck')
,(6197,'Innsbruck')
,(6198,'Innsbruck')
,(6199,'Innsbruck')
,(6201,'Bregenz'     )
,(6202,'Bregenz'     )
,(6203,'Bregenz'     )

Une requête :

SELECT DISTINCT T2.Description, 
    SUBSTRING(
        (
            SELECT ','+CAST(T1.ID AS VARCHAR)  AS [text()]
            FROM @Table T1
            WHERE T1.Description = T2.Description
            ORDER BY T1.Description
            FOR XML PATH ('')
        ), 2, 1000) [Ids]
FROM @Table T2

Résultat :

Description Ids
Bregenz     6201,6202,6203
Innsbruck   6196,6197,6198,6199
Salzburg    6192,6193,6194

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X