2010-08-26 3 views
4

У меня есть необходимость взять результаты запроса для каждой позиции и создать список материалов с разделителями с запятой, которые составляют этот элемент.SQL Server 2008 - строка конкатенации

схемы макияж:

Таблица: LineItems (Unique Пункт листинг) LineItems_Materials (многие ко многим) материалы (уникальный материал листинг)

Позиция: ID | LineItem 1 | '1A .1'

LineItems_Materials: ID | LineItemId | MaterialID 1 | 1 | 1 2 | 1 | 2 3 | 1 | 3

Материалы: ID | Материал 1 | Бетон 2 | Сталь 3 | Dirt

Так для линии Пункт 1 (1a.1) Я хочу, чтобы это показать бетон, сталь, грязь

Я знаю, что могу написать функцию для этого. Я использовал CTE в функции .... Я мог бы использовать цикл while. Есть ли другой метод, который будет лучше?

Вот что я (Сценарий будет строить объекты, данные нагрузки и создания функции):

SCRIPT: 
    IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[test].[UFN_LineItem_Materials]') AND type in (N'FN', N'IF', N'TF', N'FS', N'FT')) 
    DROP FUNCTION [test].[UFN_LineItem_Materials] 
    GO 
    IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[test].[LineItems]') AND type in (N'U')) 
    DROP TABLE [test].[LineItems] 
    GO 
    IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[test].[Materials]') AND type in (N'U')) 
    DROP TABLE [test].[Materials] 
    GO 
    IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[test].[LineItems_Materials]') AND type in (N'U')) 
    DROP TABLE [test].[LineItems_Materials] 
    GO 
    IF EXISTS (SELECT * FROM sys.schemas WHERE name = N'test') 
    DROP SCHEMA [test] 
    GO 
    CREATE SCHEMA [test] AUTHORIZATION [dbo] 
    GO 

    Create Table test.Materials(
    MaterialID INT IDENTITY(1,1), 
    Material varchar(100)); 

    Insert Into test.Materials 
    Values('Concrete'); 


    Insert Into test.Materials 
    Values('Steel'); 


    Insert Into test.Materials 
    Values('Dirt'); 

    GO 
    Create Table test.LineItems_Materials(
    LineItemMaterialID INT IDENTITY(1,1), 
    LineItemID   INT, 
    MaterialID   INT) 

    GO 

    Insert Into test.LineItems_Materials 
    Select 1,1 
    UNION 
    Select 1,2 
    UNION 
    Select 1,3 


    GO 


    CREATE TABLE [test].[LineItems](
     [LineItemID] [int] IDENTITY(1,1) NOT NULL, 
     [ItemNumber] [varchar](25) NULL 
    ) ON [PRIMARY] 

    GO 

    Insert Into [test].[LineItems] 
    Select '1A.1' 


    GO 
    ------------------------------------------------------------- 
    --Build Material Strings (;) example: List of Materials 
    ------------------------------------------------------------ 

    CREATE FUNCTION test.UFN_LineItem_Materials(@LineItemID INT) 
     RETURNS VARCHAR(100) 
    AS 

    BEGIN 

     DECLARE @Materials Varchar(100) = '' 
     ;with CTE 
     AS(
     Select lm.LineItemID,m.MaterialID,m.Material 
     from test.LineItems_Materials lm 
     inner join test.Materials m on lm.MaterialID = m.MaterialID 
     Where lm.LineItemID = @LineItemID 
     ) 
     Select @Materials += ';' + c.Material 
     from CTE c; 

     SET @Materials = substring(@Materials,2,LEN(@Materials)-1); 

     RETURN @Materials; 

    END 
    GO 

    Select lm.LineItemID,test.UFN_LineItem_Materials(lm.LineItemID) Materials 
    From test.Materials m 
    inner join test.LineItems_Materials lm on m.MaterialID = lm.MaterialID 
    Where m.Material = 'Concrete' 

Любые другие идеи?

Всегда ценим обратную связь

-S

ответ

6

Если вы хотите, чтобы сцепить значения по строкам, используйте FOR XML трюка, например:

SELECT Name + ',' 
FROM Project 
FOR XML PATH('') 

Вот более полный пример:

select LineItemID, (
     Select m.Material + ',' 
     From test.Materials m 
     inner join test.LineItems_Materials lm1 on m.MaterialID = lm1.MaterialID 
     Where m.MaterialID in (select MaterialID from test.LineItems_Materials where LineItemID = lm2.LineItemID) 
     FOR XML PATH('') 
    ) as Materials 
from test.LineItems_Materials lm2 
group by LineItemID 
+0

, что деньги ... спасибо! – scarpacci

+0

Это действительно близко к тому, что я построил – scarpacci

+0

+1 для самой простой презентации 'for xml' – DonBecker

0

Следующая TSQL будет делать трюк, нужно просто заменить соответствующие столбцы и таблицы имен

DECLARE @vString NVARCHAR(500) 
SET @vString = '' 
SELECT @vString = @vString + ColumnNameToConcatenate + ',' 
FROM TableToPickColumnFrom 

SELECT SUBSTRING(@vString, 0, LEN(@vString) -1) AS ConcatenatedText 

Он также снимите задний конец , с конца строки, чтобы получить:

Значение1, значение2, Value3

Rathe чем:

Значение1, значение2, Value3,

1

попробовать что-то вроде этого:

declare @x table (x varchar(5)) 
insert @x values ('AAAA') 
insert @x values ('BBBB') 
insert @x values ('CCCC') 

SELECT 
    STUFF(
      (
       SELECT ','+x 
        FROM @x 
        FOR XML PATH('') 

      ), 1, 1, '' 
     ) AS ColName 

ВЫВОД:

ColName 
------------------- 
AAAA,BBBB,CCCC 

(1 row(s) affected) 

или как это:

declare @x table (RowID int, x varchar(5)) 
insert @x values (1,'AAAA') 
insert @x values (1,'BBBB') 
insert @x values (1,'CCCC') 
insert @x values (2,'aa') 
insert @x values (3,'abc') 
insert @x values (3,'123') 

SELECT 
    a.RowID, 
    STUFF(
      (
       SELECT ', '+b.x 
        FROM @x b 
        WHERE a.RowID=b.RowID 
        order by b.x 
        FOR XML PATH('') 
      ), 1, 2, '' 
     ) AS ColName 
    FROM @x a 
    GROUP BY a.RowID 

ВЫХОДА:

RowID  ColName 
----------- ----------------- 
1   AAAA, BBBB, CCCC 
2   aa 
3   123, abc 

(3 row(s) affected) 
Смежные вопросы