Как переименовать узел Oracle XMLTYPE

У меня есть XMLType в PL / SQL, и мне нужно переименовать некоторые узлы и некоторые значения. Например:

<root>
   <fields>
        <a>foo</a>
        <b>bar</b>
   </fields>
</root>

Я хочу превратить вышесказанное в это:

<root>
   <fields>
        <a>foo</a>
        <c>baz</c>
   </fields>
</root>

Я знаю, что могу обновить значение следующим образом:

SELECT UpdateXML(my_xml, '/root/fields/b/text()', 'baz')
  INTO my_xml_updated
  FROM DUAL;

Результат:

<root>
   <fields>
        <a>foo</a>
        <b>baz</b>
   </fields>
</root>

Но как я могу обновить имя узла с <b> до <c> (не затрагивая содержимое узла)?


person wweicker    schedule 17.10.2011    source источник
comment
Ваша ссылка указывает на документацию Berkeley DB, а не на Oracle DB.   -  person user272735    schedule 18.10.2011
comment
Спасибо, я удалил этот аспект вопроса, поскольку он был избыточным.   -  person wweicker    schedule 19.10.2011


Ответы (2)


Один из вариантов - использовать XMLTRANSFORM для переименования узел. См. Также, например, Переименуйте узлы с помощью XSLT.

with
xmldata as (select xmltype('<root>
  <fields>
    <a>foo</a>
    <b>bar</b>
  </fields>
</root>') val from dual),
stylesheet as (select '<?xml version="1.0"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <!-- Identity transformation -->
  <xsl:template match="node()|@*">
    <xsl:copy>
      <xsl:apply-templates select="node()|@*"/>
    </xsl:copy>
  </xsl:template>
  <!-- Identity transformation overridden for element b -->
  <xsl:template match="b">
    <xsl:element name="c">
      <xsl:apply-templates select="node()|@*"/>
    </xsl:element>
  </xsl:template>
</xsl:stylesheet>' val from dual)
select xmltransform(x.val, s.val) from xmldata x, stylesheet s;

Вывод:

XMLTRANSFORM(X.VAL,S.VAL)
--------------------------------------------------------------------------------
<root>
  <fields>
    <a>foo</a>
    <c>bar</c>
  </fields>
</root>
person user272735    schedule 18.10.2011
comment
Это отбрасывает содержимое c, но может быть изменено следующим образом: ‹xsl: template match = b› ‹xsl: element name = c› ‹xsl: apply-templates select = @ * | node () /› ‹/ xsl: element ›‹ / Xsl: template › - person wweicker; 19.10.2011
comment
@wweicker: Да, но теперь ответ обновлен, чтобы соответствовать изменившимся требованиям :) - person user272735; 19.10.2011

Или вы можете определить таблицу стилей xsl, которая внесет изменения, и использовать dbms_xslprocessor для внесения ваших изменений.

person Jim Hudson    schedule 18.10.2011