SQL Server XML-DML, как заменить значение элемента значением относительного xpath - PullRequest
2 голосов
/ 20 января 2012

У меня есть столбец xml со следующими данными:

<Policy>
  <Name>
    <Id>adf</Id>
    <First>Alan</First>
    <Last>Turing</Last>
  </Name>
  <Name>
    <Id>asdf</Id>
    <Business>Vandelay Industries</Business>
  </Name>
  <Name>
    <Id>asdf</Id>
    <First>Dennis</First>
    <Last>Ritchie</Last>
  </Name>
</Policy>

Как мне написать запрос XML-DML, который заменит значение элемента Id для каждого Имени во всех строках, чтобы оно содержало значения First, Last и Business. Например:

<Policy>
  <Name>
    <Id>AlanTuring</Id>
    <First>Alan</First>
    <Last>Turing</Last>
  </Name>
  <Name>
    <Id>Vandelay Industries</Id>
    <Business>Vandelay Industries</Business>
  </Name>
  <Name>
    <Id>DennisRitchie</Id>
    <First>Dennis</First>
    <Last>Ritchie</Last>
  </Name>
</Policy>

Вот моя неудачная попытка:

update policy set data.modify('
  replace value of (//Name/Id/text())[1] 
  with concat(
    (//Name/First/text())[1], 
    (//Name/Last/text())[1], 
    (//Name/Business/text())[1])') 

Который производит следующее:

<Policy>
  <Name>
    <Id>AlanTuringVandelay Industries</Id>
    <First>Alan</First>
    <Last>Turing</Last>
  </Name>
  <Name>
    <Id>asdf</Id>
    <Business>Vandelay Industries</Business>
  </Name>
  <Name>
    <Id>asdf</Id>
    <First>Dennis</First>
    <Last>Ritchie</Last>
  </Name>
</Policy>

Ответы [ 2 ]

3 голосов
/ 20 января 2012

Это работает.

declare @i int
declare @maxNames int
set @i = 1
set @maxNames = (select max(data.value('count(//Name)', 'int')) from Policy)
while @i <= @maxNames begin
    update policy set data.modify('
      replace value of (//Name[sql:variable("@i")]/Id/text())[1] 
      with concat(
        (//Name[sql:variable("@i")]/First/text())[1], 
        (//Name[sql:variable("@i")]/Last/text())[1], 
        (//Name[sql:variable("@i")]/Business/text())[1])
    ') 
    set @i = @i + 1
end
1 голос
/ 24 ноября 2015

Опция на основе набора будет такой:

    update policy 
    set data.modify('
    replace value of (//Name[sql:column("id")]/Id/text())[1]
    with concat(
    (//Name[sql:column("id")]/First/text())[1],
    (//Name[sql:column("id")]/Last/text())[1],
    (//Name[sql:column("id")]/Business/text())[1])
    ') 
...