Как мне получить:
id       Name       Value
1          A          4
1          B          8
2          C          9
к
id          Column
1          A:4, B:8
2          C:9
Как мне получить:
id       Name       Value
1          A          4
1          B          8
2          C          9
к
id          Column
1          A:4, B:8
2          C:9
Не требуется CURSOR, WHILE или пользовательская функция.
Просто нужно быть творческим с FOR XML и PATH.
[Примечание. Это решение работает только с SQL 2005 и более поздними версиями. Исходный вопрос не указывал используемую версию.]
CREATE TABLE #YourTable ([ID] INT, [Name] CHAR(1), [Value] INT)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'A',4)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'B',8)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (2,'C',9)
SELECT 
  [ID],
  STUFF((
    SELECT ', ' + [Name] + ':' + CAST([Value] AS VARCHAR(MAX)) 
    FROM #YourTable 
    WHERE (ID = Results.ID) 
    FOR XML PATH(''),TYPE).value('(./text())[1]','VARCHAR(MAX)')
  ,1,2,'') AS NameValues
FROM #YourTable Results
GROUP BY ID
DROP TABLE #YourTable
Если это SQL Server 2017 или SQL Server Vnext, SQL Azure вы можете использовать string_agg, как показано ниже:
select id, string_agg(concat(name, ':', [value]), ', ')
    from #YourTable 
    group by id
использование XML-пути не будет идеально конкатенировать, как вы могли ожидать... он заменит "&" с "& amp;" и также будет беспорядок с <" and ">
... может быть, несколько других вещей, не уверен... но вы можете попробовать это
Я столкнулся с обходным решением для этого... вам нужно заменить:
FOR XML PATH('')
)
с:
FOR XML PATH(''),TYPE
).value('(./text())[1]','VARCHAR(MAX)')
... или NVARCHAR(MAX), если это то, что вы используете.
почему, черт возьми, SQL не имеет агрегированной функции сцепления? это PITA.
 Я столкнулся с парой проблем, когда попытался преобразовать предложение Кевина Фэйрчайлда для работы со строками, содержащими пробелы и специальные символы XML (&, <, >), которые были закодированы.
Окончательная версия моего кода (которая не отвечает на первоначальный вопрос, но может быть полезна кому-то) выглядит следующим образом:
CREATE TABLE #YourTable ([ID] INT, [Name] VARCHAR(MAX), [Value] INT)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'Oranges & Lemons',4)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'1 < 2',8)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (2,'C',9)
SELECT  [ID],
  STUFF((
    SELECT ', ' + CAST([Name] AS VARCHAR(MAX))
    FROM #YourTable WHERE (ID = Results.ID) 
    FOR XML PATH(''),TYPE 
     /* Use .value to uncomment XML entities e.g. > < etc*/
    ).value('.','VARCHAR(MAX)') 
  ,1,2,'') as NameValues
FROM    #YourTable Results
GROUP BY ID
DROP TABLE #YourTable
 Вместо использования пробела в качестве разделителя и замены всех пробелов запятыми, он просто добавляет запятую и пробел к каждому значению, а затем использует STUFF для удаления первых двух символов.
Кодировка XML обеспечивается автоматически с помощью директивы TYPE.
Другая опция, использующая Sql Server 2005 и выше
---- test data
declare @t table (OUTPUTID int, SCHME varchar(10), DESCR varchar(10))
insert @t select 1125439       ,'CKT','Approved'
insert @t select 1125439       ,'RENO','Approved'
insert @t select 1134691       ,'CKT','Approved'
insert @t select 1134691       ,'RENO','Approved'
insert @t select 1134691       ,'pn','Approved'
---- actual query
;with cte(outputid,combined,rn)
as
(
  select outputid, SCHME + ' ('+DESCR+')', rn=ROW_NUMBER() over (PARTITION by outputid order by schme, descr)
  from @t
)
,cte2(outputid,finalstatus,rn)
as
(
select OUTPUTID, convert(varchar(max),combined), 1 from cte where rn=1
union all
select cte2.outputid, convert(varchar(max),cte2.finalstatus+', '+cte.combined), cte2.rn+1
from cte2
inner join cte on cte.OUTPUTID = cte2.outputid and cte.rn=cte2.rn+1
)
select outputid, MAX(finalstatus) from cte2 group by outputid
Установите агрегаты SQLCLR из http://groupconcat.codeplex.com
Затем вы можете написать такой код, чтобы получить результат, который вы просили:
CREATE TABLE foo
(
 id INT,
 name CHAR(1),
 Value CHAR(1)
);
INSERT  INTO dbo.foo
    (id, name, Value)
VALUES  (1, 'A', '4'),
        (1, 'B', '8'),
        (2, 'C', '9');
SELECT  id,
    dbo.GROUP_CONCAT(name + ':' + Value) AS [Column]
FROM    dbo.foo
GROUP BY id;
SQL Server 2005 и более поздние версии позволяют создавать собственные настраиваемые функции агрегации, в том числе для таких вещей, как конкатенация - см. образец в нижней части связанная статья.
В Oracle вы можете использовать агрегатную функцию LISTAGG.
Оригинальные записи
name   type
------------
name1  type1
name2  type2
name2  type3
SQL
SELECT name, LISTAGG(type, '; ') WITHIN GROUP(ORDER BY name)
FROM table
GROUP BY name
Результат в
name   type
------------
name1  type1
name2  type2; type3
Спустя восемь лет... Microsoft SQL Server vNext Database Engine наконец-то расширил Transact-SQL, чтобы напрямую поддерживать сгруппированную конкатенацию строк. В версии 5.0 Технического просмотра сообщества добавлена функция STRING_AGG, а CTP 1.1 добавила предложение WITHIN GROUP для функции STRING_AGG.
Ссылка: https://msdn.microsoft.com/en-us/library/mt775028.aspx
Этот вопрос задается здесь очень часто, и решение будет сильно зависеть от основных требований:
https://stackoverflow.com/search?q=sql+pivot
и
https://stackoverflow.com/search?q=sql+concatenate
Как правило, для этого не существует SQL-only без динамического sql, пользовательской функции или курсора.
Просто чтобы добавить к тому, что сказал Кейд, это, как правило, внешний вид дисплея, и поэтому его следует обрабатывать. Я знаю, что иногда проще писать что-то 100% в SQL для таких вещей, как экспорт файлов или другие "только SQL-решения", но большую часть времени это конкатенация следует обрабатывать на вашем уровне отображения.
Это просто дополнение к посту Кевина Фэрчайлд (кстати, очень умный). Я бы добавил его в качестве комментария, но пока у меня недостаточно очков:)
Я использовал эту идею для представления, над которым я работал, но элементы, которые я контактировал, содержали пробелы. Поэтому я немного изменил код, чтобы не использовать пробелы в качестве разделителей.
Снова спасибо за крутое обходное решение Кевина!
CREATE TABLE #YourTable ( [ID] INT, [Name] CHAR(1), [Value] INT ) 
INSERT INTO #YourTable ([ID], [Name], [Value]) VALUES (1, 'A', 4) 
INSERT INTO #YourTable ([ID], [Name], [Value]) VALUES (1, 'B', 8) 
INSERT INTO #YourTable ([ID], [Name], [Value]) VALUES (2, 'C', 9) 
SELECT [ID], 
       REPLACE(REPLACE(REPLACE(
                          (SELECT [Name] + ':' + CAST([Value] AS VARCHAR(MAX)) as A 
                           FROM   #YourTable 
                           WHERE  ( ID = Results.ID ) 
                           FOR XML PATH (''))
                        , '</A><A>', ', ')
                ,'<A>','')
        ,'</A>','') AS NameValues 
FROM   #YourTable Results 
GROUP  BY ID 
DROP TABLE #YourTable 
Не нужно курсора... достаточно цикла while.
------------------------------
-- Setup
------------------------------
DECLARE @Source TABLE
(
  id int,
  Name varchar(30),
  Value int
)
DECLARE @Target TABLE
(
  id int,
  Result varchar(max) 
)
INSERT INTO @Source(id, Name, Value) SELECT 1, 'A', 4
INSERT INTO @Source(id, Name, Value) SELECT 1, 'B', 8
INSERT INTO @Source(id, Name, Value) SELECT 2, 'C', 9
------------------------------
-- Technique
------------------------------
INSERT INTO @Target (id)
SELECT id
FROM @Source
GROUP BY id
DECLARE @id int, @Result varchar(max)
SET @id = (SELECT MIN(id) FROM @Target)
WHILE @id is not null
BEGIN
  SET @Result = null
  SELECT @Result =
    CASE
      WHEN @Result is null
      THEN ''
      ELSE @Result + ', '
    END + s.Name + ':' + convert(varchar(30),s.Value)
  FROM @Source s
  WHERE id = @id
  UPDATE @Target
  SET Result = @Result
  WHERE id = @id
  SET @id = (SELECT MIN(id) FROM @Target WHERE @id < id)
END
SELECT *
FROM @Target
Давайте получим очень просто:
SELECT stuff(
    (
    select ', ' + x from (SELECT 'xxx' x union select 'yyyy') tb 
    FOR XML PATH('')
    )
, 1, 2, '')
Заменить эту строку:
select ', ' + x from (SELECT 'xxx' x union select 'yyyy') tb
С вашим запросом.
не видел каких-либо кросс-приложений, также нет необходимости в извлечении xml. Вот немного другая версия того, что написал Кевин Фэрчайлд. Это быстрее и проще в использовании в более сложных запросах:
   select T.ID
,MAX(X.cl) NameValues
 from #YourTable T
 CROSS APPLY 
 (select STUFF((
    SELECT ', ' + [Name] + ':' + CAST([Value] AS VARCHAR(MAX))
    FROM #YourTable 
    WHERE (ID = T.ID) 
    FOR XML PATH(''))
  ,1,2,'')  [cl]) X
  GROUP BY T.ID
Вы можете повысить производительность значительно следующим образом, если группа содержит в основном один элемент:
SELECT 
  [ID],
CASE WHEN MAX( [Name]) = MIN( [Name]) THEN 
MAX( [Name]) NameValues
ELSE
  STUFF((
    SELECT ', ' + [Name] + ':' + CAST([Value] AS VARCHAR(MAX)) 
    FROM #YourTable 
    WHERE (ID = Results.ID) 
    FOR XML PATH(''),TYPE).value('(./text())[1]','VARCHAR(MAX)')
  ,1,2,'') AS NameValues
END
FROM #YourTable Results
GROUP BY ID
Использование функции замены и FOR JSON PATH
SELECT T3.DEPT, REPLACE(REPLACE(T3.ENAME,'{"ENAME":"',''),'"}','') AS ENAME_LIST
FROM (
 SELECT DEPT, (SELECT ENAME AS [ENAME]
        FROM EMPLOYEE T2
        WHERE T2.DEPT=T1.DEPT
        FOR JSON PATH,WITHOUT_ARRAY_WRAPPER) ENAME
    FROM EMPLOYEE T1
    GROUP BY DEPT) T3
Для ознакомления с примерами данных и другими способами нажмите здесь