Я ищу способ обновления (.modify ('insert into ..'), чтобы быть конкретным) столбец XML в таблице с фрагментами xml из другой таблицы связанных внешним ключом.T-SQL XML Update с XML-фрагментом из другой связанной таблицы
Например, моя структура таблицы выглядит, как показано ниже (упрощенный):
- полей от 1 до 5 в каждой таблице может быть проигнорирована; их единственная цель - объяснить поля Xx.
- Поля Xx в каждой таблице определяются как XML и предварительно заполнены фрагментом XML, который содержит поля из таблицы внутри тега, имя таблицы. После этого списка отображаются фрагменты XML.
- Таблица B и таблица C имеют внешний ключ FK_A, который связывает их с таблицей A. Таблица A - B и таблица A - C являются одними для многих (одна запись в A может иметь несколько записей в B и C).
Теперь, к значениям выборки поля Хх до того, что мне нужно достичь:
<!-- Table A record 1 -->
<TableA PK_A="1">
<A1>Avalue</A1>
<A2>Avalue</A2>
<A3>Avalue</A3>
<A4>Avalue</A4>
<A5>Avalue</A5>
</TableA>
<!-- Table B record 1 -->
<TableB PK_B="1" FK_A="1">
<B1>Bvalue1</B1>
<B2>Bvalue1</B2>
<B3>Bvalue1</B3>
<B4>Bvalue1</B4>
<B5>Bvalue1</B5>
</TableB>
<!-- Table B record 2 -->
<TableB PK_B="2" FK_A="1">
<B1>Bvalue2</B1>
<B2>Bvalue2</B2>
<B3>Bvalue2</B3>
<B4>Bvalue2</B4>
<B5>Bvalue2</B5>
</TableB>
<!-- Table C record 1 -->
<TableC PK_C="1" FK_A="1">
<C1>Cvalue1</C1>
<C2>Cvalue1</C2>
<C3>Cvalue1</C3>
<C4>Cvalue1</C4>
<C5>Cvalue1</C5>
</TableC>
<!-- Table C record 2 -->
<TableC PK_C="2" FK_A="1">
<C1>Cvalue2</C1>
<C2>Cvalue2</C2>
<C3>Cvalue2</C3>
<C4>Cvalue2</C4>
<C5>Cvalue2</C5>
</TableC>
Проблема здесь в том, как бы я обновить таблицу A, вставляя все XBS и XCS как первый (или последний) в соответствующий XA? Я предпочитаю одну операцию для обновления всей таблицы.
После операции XA должен выглядеть следующим образом:
<!-- Table A record 1 -->
<TableA PK_A="1">
<!-- Table B record 1 -->
<TableB PK_B="1" FK_A="1">
<B1>Bvalue1</B1>
<B2>Bvalue1</B2>
<B3>Bvalue1</B3>
<B4>Bvalue1</B4>
<B5>Bvalue1</B5>
</TableB>
<!-- Table B record 2 -->
<TableB PK_B="2" FK_A="1">
<B1>Bvalue2</B1>
<B2>Bvalue2</B2>
<B3>Bvalue2</B3>
<B4>Bvalue2</B4>
<B5>Bvalue2</B5>
</TableB>
<!-- Table C record 1 -->
<TableC PK_C="1" FK_A="1">
<C1>Cvalue1</C1>
<C2>Cvalue1</C2>
<C3>Cvalue1</C3>
<C4>Cvalue1</C4>
<C5>Cvalue1</C5>
</TableC>
<!-- Table C record 2 -->
<TableC PK_C="2" FK_A="1">
<C1>Cvalue2</C1>
<C2>Cvalue2</C2>
<C3>Cvalue2</C3>
<C4>Cvalue2</C4>
<C5>Cvalue2</C5>
</TableC>
<A1>Avalue</A1>
<A2>Avalue</A2>
<A3>Avalue</A3>
<A4>Avalue</A4>
<A5>Avalue</A5>
</TableA>
Что я пытался до сих пор?
Мое лучшее лидерство до сих пор было подходом к разделению и правительству (решение из двух частей, которое не работает).
WITH CTEB (PK_A, XA, XB)
AS (SELECT A.PK_A,
a.XA,
b.XB
FROM TableA AS a
INNER JOIN
TableB AS b
ON b.FK_A = a.PK_A)
UPDATE CTEB
SET XA.modify('insert sql:column("XB") as last into (/TableA)[1]');
WITH CTEC (PK_A, XA, XC)
AS (SELECT A.PK_A,
a.XA,
c.XC
FROM TableA AS a
INNER JOIN
TableC AS c
ON c.FK_A = a.PK_A)
UPDATE CTEC
SET XA.modify('insert sql:column("XC") as last into (/TableA)[1]');
Edit:
Извиняюсь за не давая каких-либо текстовых значений для копирования вставки. Вот так.
DECLARE @A TABLE (PK_A INT, XA XML);
DECLARE @B TABLE (PK_B INT, XB XML, FK_A INT);
DECLARE @C TABLE (PK_C INT, XC XML, FK_A INT);
INSERT INTO @A
VALUES (1, '<TableA PK_A="1"><A1>Avalue</A1><A2>Avalue</A2><A3>Avalue</A3><A4>Avalue</A4><A5>Avalue</A5></TableA>')
INSERT INTO @A
VALUES (2, '<TableA PK_A="2"><A1>Avalue</A1><A2>Avalue</A2><A3>Avalue</A3><A4>Avalue</A4><A5>Avalue</A5></TableA>')
INSERT INTO @B
VALUES (1,'<TableB PK_B="1" FK_A="1"><B1>Bvalue1</B1><B2>Bvalue1</B2><B3>Bvalue1</B3><B4>Bvalue1</B4><B5>Bvalue1</B5></TableB>', 1)
INSERT INTO @B
VALUES (2, '<TableB PK_B="2" FK_A="1"><B1>Bvalue2</B1><B2>Bvalue2</B2><B3>Bvalue2</B3><B4>Bvalue2</B4><B5>Bvalue2</B5></TableB>', 1)
INSERT INTO @B
VALUES (3, '<TableB PK_B="3" FK_A="2"><B1>Bvalue3</B1><B2>Bvalue3</B2><B3>Bvalue3</B3><B4>Bvalue3</B4><B5>Bvalue3</B5></TableB>', 2)
INSERT INTO @B
VALUES (4, '<TableB PK_B="4" FK_A="2"><B1>Bvalue4</B1><B2>Bvalue4</B2><B3>Bvalue4</B3><B4>Bvalue4</B4><B5>Bvalue4</B5></TableB>', 2)
INSERT INTO @C
VALUES (1, '<TableC PK_C="1" FK_A="1"><C1>Cvalue1</C1><C2>Cvalue1</C2><C3>Cvalue1</C3><C4>Cvalue1</C4><C5>Cvalue1</C5></TableC>', 1)
INSERT INTO @C
VALUES (2, '<TableC PK_C="2" FK_A="1"><C1>Cvalue2</C1><C2>Cvalue2</C2><C3>Cvalue2</C3><C4>Cvalue2</C4><C5>Cvalue2</C5></TableC>', 1)
INSERT INTO @C
VALUES (3, '<TableC PK_C="3" FK_A="2"><C1>Cvalue3</C1><C2>Cvalue3</C2><C3>Cvalue3</C3><C4>Cvalue3</C4><C5>Cvalue3</C5></TableC>', 2)
INSERT INTO @C
VALUES (4, '<TableC PK_C="4" FK_A="2"><C1>Cvalue4</C1><C2>Cvalue4</C2><C3>Cvalue4</C3><C4>Cvalue4</C4><C5>Cvalue4</C5></TableC>', 2);
WITH CTEB (PK_A, XA, XB)
AS (SELECT A.PK_A,
a.XA,
b.XB
FROM @A AS a, @B as b
WHERE b.FK_A = a.PK_A)
UPDATE CTEB
SET XA.modify('insert sql:column("XB") as first into (/TableA)[1]');
WITH CTEC (PK_A, XA, XC)
AS (SELECT A.PK_A,
a.XA,
c.XC
FROM @A AS a
INNER JOIN
@C AS c
ON c.FK_A = a.PK_A)
UPDATE CTEC
SET XA.modify('insert sql:column("XC") as first into (/TableA)[1]');
SELECT * FROM @A;
Это обновляет XML только с первого фрагмента XML из таблицы B и C. Таблица
Update: Для тех, кто может прийти через этот вопрос, пожалуйста, посмотрите на @Shnugo «s ответьте вместе с заметным ответом. Оба подхода идеальны. Я отметил решение @ gofr1 как ответ просто потому, что он первым ответил. Еще одно соображение для будущих охотников за поисками заключается в том, нравится ли вам CTE или sub-select (как указал Шнуго).
В будущем, пожалуйста, добавьте XML-фрагменты непосредственно. Не делайте скриншот и публикуйте это. Люди хотят скопировать/вставить эти xml-фрагменты, чтобы поиграть с ними, они предпочитают не вводить их сами. GL. –
Простите, добавили образец структуры таблицы и данные для работы. – Jayachandran
Не требуется никаких извинений :). Это намек для вас; люди будут более охотно помогать, если им не придется набирать пример xml-фрагмента, чтобы начать работу над ответом. –