Конкатенация строк T-SQL из строк - PullRequest
0 голосов
/ 14 ноября 2011

Я пытаюсь обновить столбец во временной таблице, объединяя строки из другой таблицы.

Посмотрите на это:

DECLARE @Test VARCHAR(MAX);

CREATE TABLE #Test
(
     EmployeeId INT
     ,Html VARCHAR(MAX)
);

CREATE TABLE #EmployeeItems
(
    EmployeeId INT
    ,ItemNo INT
);

INSERT INTO #EmployeeItems (EmployeeId, ItemNo)
VALUES
    (1, 1)
    ,(1, 2);

INSERT INTO #Test (EmployeeId, Html) VALUES (1, '<div class="first">');
SET @Test = '<div class="first">';

UPDATE T SET Html += '<div class="second">' + CAST(E.ItemNo AS VARCHAR) + '</div>'
FROM #Test AS T
    JOIN #EmployeeItems AS E ON T.EmployeeId = E.EmployeeId;

SELECT @Test += '<div class="second">' + CAST(E.ItemNo AS VARCHAR) + '</div>'
FROM #Test AS T
JOIN #EmployeeItems AS E ON T.EmployeeId = E.EmployeeId;

UPDATE #Test SET Html += '</div>';
SET @Test += '</div>';

SELECT Html FROM #Test;
SELECT @Test;

DROP TABLE #Test;
DROP TABLE #EmployeeItems;

Столбец в таблице #Test содержит: <div class="first"><div class="second">1</div></div>

А переменная @Test содержит: <div class="first"><div class="second">1</div><div class="second">2</div></div>

Почему это? Какая разница? Я думал, что получу те же результаты, но в случае таблицы он объединяет только одну строку, первую.

Что мне вместо этого сделать, чтобы обновить таблицу без курсора?

EDIT

Моя первоначальная проблема возникла из-за этого:

Мои входные данные

CREATE TABLE #Actions(EmployeeId INT,EmployeeName VARCHAR(100),ActionStart TIME,ActionEnd TIME,Type VARCHAR(10));
INSERT INTO #Actions(EmployeeId,EmployeeName,ActionStart,ActionEnd, Type)
VALUES (1,'Bob','09:00','12:00', 'action'),(1,'Bob','14:30','16:00', 'action'),(1,'Bob','18:00','20:00', 'event'),(2,'Susan','10:00','12:00', 'action');

Я хочу вывод, как этот

<div class="employee" employeeid="1" employeename="Bob">
    <div class="action" start="09:00" end="12:00" type="action"></div>
    <div class="action" start="14:30" end="16:00" type="action"></div>
    <div class="action" start="18:00" end="20:00" type="event"></div>
</div>
<div class="employee" employeeid="2" employeename="Susan">
    <div class="action" start="10:00" end="12:00" type="action"></div>
</div>

Как и в первом примере, это упрощенный пример. Но если я решу дело, я смогу решить свою проблему. Как бы вы сделали это с предложением FOR XML?

Ответы [ 2 ]

0 голосов
/ 14 ноября 2011

Думайте о конкатенации переменных как о странности;это неподдерживаемое поведение, которое часто используется в коде :) Если вы хотите продолжать использовать его, вы можете изменить свой оператор UPDATE следующим образом:

UPDATE t 
SET HTML = @Test
FROM #test t

Альтернативным методом будет использование синтаксиса FOR XML дляconcatenate

CREATE TABLE #Test
    (
      EmployeeId INT
    , Html VARCHAR(MAX)
    ) ;
CREATE TABLE #EmployeeItems
    (
      EmployeeId INT
    , ItemNo INT
    ) ;
INSERT  INTO #EmployeeItems
        ( EmployeeId, ItemNo )
VALUES  ( 1, 1 )     ,
        ( 1, 2 ) ;
INSERT  INTO #Test
        ( EmployeeId, Html )
VALUES  ( 1, '<div class="first">' ) ;

UPDATE  #Test
SET     HTML+= REPLACE(REPLACE(( SELECT '|div class="second"|'
                                        + CAST(E.ItemNo AS VARCHAR) + '|/div|'
                                 FROM   #Test AS T
                                        JOIN #EmployeeItems AS E ON T.EmployeeId = E.EmployeeId
                               FOR
                                 XML PATH('')
                               ), '|div class="second"|',
                               '<div class="second">'), '|/div|', '</div>')    
UPDATE  #Test
SET     Html += '</div>' ;

SELECT  Html
FROM    #Test ;


DROP TABLE #Test ;
DROP TABLE #EmployeeItems ;  

Или (и, возможно, лучше), просто создайте свой XML-документ напрямую.

SELECT  'first' AS "div/@class"
      , 'second' AS "div/div/@class"
      , e.ItemNo AS "div/div"
FROM    #EmployeeItems e
FOR     XML PATH('') 

Дополнительный ответ для отредактированного вопроса:

BEGIN TRAN


CREATE TABLE #Actions
    (
      EmployeeId INT
    , EmployeeName VARCHAR(100)
    , ActionStart TIME
    , ActionEnd TIME
    , Type VARCHAR(10)
    ) ;
INSERT  INTO #Actions
        ( EmployeeId, EmployeeName, ActionStart, ActionEnd, Type )
VALUES  ( 1, 'Bob', '09:00', '12:00', 'action' ),
        ( 1, 'Bob', '14:30', '16:00', 'action' ),
        ( 1, 'Bob', '18:00', '20:00', 'event' ),
        ( 2, 'Susan', '10:00', '12:00', 'action' ) ; 

;
WITH    CTE
          AS ( SELECT DISTINCT
                        EmployeeID
                      , EmployeeName
               FROM     #actions
             )
    SELECT  'employee' AS "@class"
          , employeeid AS "@employeeid"
          , employeename AS "@employeename"
          , ( SELECT    'action' AS "@class"
                      , ActionStart AS "@start"
                      , ActionEnd AS "@end"
                      , Type AS "@type"
              FROM      #Actions a2
              WHERE     a2.EmployeeId = a.EmployeeID
            FOR
              XML PATH('div')
                , TYPE
            )
    FROM    cte a
FOR     XML PATH('div') 


ROLLBACK TRAN       
0 голосов
/ 14 ноября 2011

Для тестовой таблицы обновление выполняется для каждой строки, поэтому, если вы выберете все строки в таблице, вы получите две строки:

<div class="first"><div class="second">1</div></div>
<div class="first"><div class="second">2</div></div>

Во втором случае '<div class="second">' + CAST(E.ItemNo AS VARCHAR) + '</div>' будет добавлено столько раз, сколько строк в таблице. Поскольку у вас есть две строки, он добавит два переменных в переменную.

Если вы хотите, чтобы каждая строка таблицы имела значение @Test, вам не нужен курсор. Просто обновите таблицу, чтобы она имела значение @Test.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...