다음을 통해 공유


replace value of(XML DML)

문서의 노드 값을 업데이트합니다.

구문

replace value of 
      Expression1 
with
      Expression2

인수

  • Expression1
    업데이트할 값이 있는 노드를 식별합니다. 하나의 노드만 식별해야 합니다. 즉 Expression1이 정적 단일 항목이어야 합니다. XML이 형식화되는 경우 노드 유형은 단순 유형이어야 합니다. 노드를 여러 개 선택하면 오류가 발생합니다. Expression1이 빈 시퀀스를 반환하면 값이 대체되지 않고 오류가 반환되지 않습니다. Expression1은 단순 유형의 콘텐츠(목록 또는 원자성 유형), 텍스트 노드 또는 특성 노드를 가진 단일 요소를 반환해야 합니다. 또한 Expression1은 공용 구조체 유형, 복합 유형, 처리 명령, 문서 노드 또는 주석 노드는 될 수 없습니다. 이들 중 하나인 경우 오류가 반환됩니다.

  • Expression2
    노드의 새 값을 식별합니다. **data()**를 암시적으로 사용하므로 단순 유형의 노드를 반환하는 식이 될 수 있습니다. 값이 값 목록일 경우 update 문은 이전 값을 목록으로 대체합니다. 형식화된 XML 인스턴스를 수정하는 경우 Expression2 유형이 Expression1과 같은 유형이거나 그 하위 유형이어야 합니다. 그렇지 않으면 오류가 반환됩니다. 형식화되지 않은 XML 인스턴스를 수정할 경우 Expression2는 원자화될 수 있는 식이어야 합니다. 그렇지 않으면 오류가 반환됩니다.

replace value of XML DML 문의 다음 예에서는 XML 문서에서 노드를 업데이트하는 방법을 보여 줍니다.

1. XML 인스턴스에서 값 바꾸기

다음 예에서는 먼저 문서 인스턴스가 xml 유형의 변수에 할당됩니다. 그러면 replace value of XML DML 문이 문서의 값을 업데이트합니다.

DECLARE @myDoc xml
SET @myDoc = '<Root>
<Location LocationID="10" 
            LaborHours="1.1"
            MachineHours=".2" >Manufacturing steps are described here.
<step>Manufacturing step 1 at this work center</step>
<step>Manufacturing step 2 at this work center</step>
</Location>
</Root>'
SELECT @myDoc

-- update text in the first manufacturing step
SET @myDoc.modify('
  replace value of (/Root/Location/step[1]/text())[1]
  with     "new text describing the manu step"
')
SELECT @myDoc
-- update attribute value
SET @myDoc.modify('
  replace value of (/Root/Location/@LaborHours)[1]
  with     "100.0"
')
SELECT @myDoc

업데이트될 대상은 경로 식에서 식 끝에 "[1]"을 추가하여 명시적으로 지정된 최대 하나의 노드여야 합니다.

2. if 식을 사용하여 대체 값 결정

아래 예에서 볼 수 있듯이 replace value of XML DML 문의 Expression2에서 if 식을 지정할 수 있습니다. Expression1은 첫 번째 작업 센터의 LaborHours 특성이 업데이트될 것임을 확인합니다. Expression2는 if 식을 사용하여 LaborHours 특성의 새 값을 확인합니다.

DECLARE @myDoc xml
SET @myDoc = '<Root>
<Location LocationID="10" 
            LaborHours=".1"
            MachineHours=".2" >Manu steps are described here.
<step>Manufacturing step 1 at this work center</step>
<step>Manufacturing step 2 at this work center</step>
</Location>
</Root>'
--SELECT @myDoc

SET @myDoc.modify('
  replace value of (/Root/Location[1]/@LaborHours)[1]
  with (
       if (count(/Root/Location[1]/step) > 3) then
         "3.0"
       else
          "1.0"
      )
')
SELECT @myDoc

3. 형식화되지 않은 XML 열에 저장된 XML 업데이트

다음 예에서는 열에 저장되어 있는 XML을 업데이트합니다.

drop table T
go
CREATE TABLE T (i int, x xml)
go
INSERT INTO T VALUES(1,'<Root>
<ProductDescription ProductID="1" ProductName="Road Bike">
<Features>
  <Warranty>1 year parts and labor</Warranty>
  <Maintenance>3 year parts and labor extended maintenance is available</Maintenance>
</Features>
</ProductDescription>
</Root>')
go
-- verify the current <ProductDescription> element
SELECT x.query(' /Root/ProductDescription')
FROM T
-- update the ProductName attribute value
UPDATE T
SET x.modify('
  replace value of (/Root/ProductDescription/@ProductName)[1]
  with "New Road Bike" ')
-- verify the update
SELECT x.query(' /Root/ProductDescription')
FROM T

4. 형식화된 XML 열에 저장되어 있는 XML 업데이트

이 예에서는 형식화된 XML 열에 저장되어 있는 제조 지침 문서의 값을 대체합니다.

이 예에서는 먼저 AdventureWorks 데이터베이스에서 형식화된 XML 열을 가진 테이블(T)을 만듭니다. 그런 다음 ProductModel 테이블의 Instructions 열의 제조 지침 XML 인스턴스를 테이블 T로 복사합니다. 그러면 테이블 T의 XML에 삽입 내용이 적용됩니다.

use AdventureWorks
go
drop table T
go
create table T(ProductModelID int primary key, 
Instructions xml (Production.ManuInstructionsSchemaCollection))
go
insert  T 
select ProductModelID, Instructions
from Production.ProductModel
where ProductModelID=7
go
--insert a new location - <Location 1000/>. 
update T
set Instructions.modify('
  declare namespace MI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
insert <MI:Location LocationID="1000"  LaborHours="1000"  LotSize="1000" >
           <MI:step>Do something using <MI:tool>hammer</MI:tool></MI:step>
         </MI:Location>
  as first
  into   (/MI:root)[1]
')
go
select Instructions
from T
go
-- Now replace manu. tool in location 1000
update T
set Instructions.modify('
  declare namespace MI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
  replace value of (/MI:root/MI:Location/MI:step/MI:tool)[1] 
  with   "screwdriver"
')
go
select Instructions
from T
-- Now replace value of lot size
update T
set Instructions.modify('
  declare namespace MI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
  replace value of (/MI:root/MI:Location/@LotSize)[1] 
  with   500 cast as xs:decimal ?
')
go
select Instructions
from T

LotSize 값을 대체할 경우 cast를 사용합니다. 값이 특정 유형인 경우 이렇게 해야 합니다. 이 예에서 값이 500인 경우 명시적 캐스트가 필요하지 않습니다.