Sqlserver
 sql >> Baza danych >  >> RDS >> Sqlserver

Połącz wiele rzędów w jeden rząd

Nie możesz mieć takiej dynamicznej liczby kolumn, ale możesz konkatenować swoje dane na ciąg:

select
    s.ItemID, s.Code, s.Name,
    stuff(
        (
            select ', ' + CAST(sv.SearchValueID AS VARCHAR)
            from ItemSearch as sv
            where sv.ItemID = s.ItemID
            for xml path(''), type
        ).value('.', 'nvarchar(128)')
    , 1, 2, '') as SearchValues
from Item as s;

lub możesz przestawić wiersze za pomocą PIVOT command lub ręcznie (wolę to drugie podejście, wydaje mi się po prostu bardziej elastyczne, ale pivot może znacznie zmniejszyć ilość kodu w określonych sytuacjach ):

with cte as (
    select
        *,
        row_number() over(partition by sv.ItemID order by sv.SearchValueID) as row_num
    from ItemSearch as sv
)
select
    s.ItemID, s.Code, s.Name,
    max(case when sv.row_num = 1 then sv.SearchValueID end) as SearchValueID1,
    max(case when sv.row_num = 2 then sv.SearchValueID end) as SearchValueID2,
    max(case when sv.row_num = 3 then sv.SearchValueID end) as SearchValueID3,
    max(case when sv.row_num = 4 then sv.SearchValueID end) as SearchValueID4
from Item as s
    inner join cte as sv on sv.ItemID = s.ItemID
group by s.ItemID, s.Code, s.Name

Możesz także zmienić poprzednią instrukcję w dynamiczny SQL tak:

declare @stmt nvarchar(max)

select
    @stmt = 
        isnull(@stmt + ',','') + 
        'max(case when sv.row_num = ' + cast(rn as nvarchar(max)) +
        ' then sv.SearchValueID end) as SearchValueID' + cast(rn as nvarchar(max))
from (
    select distinct row_number() over(partition by ItemID order by SearchValueID) as rn
    from ItemSearch
) as a

select @stmt = '
    with cte as (
        select
            *,
            row_number() over(partition by sv.ItemID order by sv.SearchValueID) as row_num
        from ItemSearch as sv
    )
    select
        s.ItemID, s.Code, s.Name,' + @stmt + '
    from Item as s
        inner join cte as sv on sv.ItemID = s.ItemID
    group by s.ItemID, s.Code, s.Name;'

exec dbo.sp_executesql @stmt = @stmt

demonstracja skrzypiec sql




  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. Przechowywana procedura parsowania ciągu

  2. Jak przekonwertować kolumnę tekstową na datę i godzinę w SQL?

  3. serwer sql wyświetla brakujące daty

  4. Nie można zapisać więcej niż 43679 znaków w kolumnie typu danych tekstowych w SQL Server

  5. Oświadczenia profilowania wewnątrz funkcji zdefiniowanej przez użytkownika