7

我在使用 Oracle SQL 和 XML 时遇到问题。

我将从外部系统获得大量格式良好的 XML 数据,用于解析、解释和填充一些表。我使用 XMLTable 编写了一个解决方案,该解决方案在表上的一个视图中进行了布局,其中包含有问题的 xml clob 列以及一些审计信息和内容(我想保持这种方式)。

命名空间让我做噩梦。显然我不能把它们放在 xmlnamespaces 子句中,因为我永远不知道它们会是什么。荒谬!相同类型的交付项目在不同时间点可能具有不同的命名空间。没有有限的清单。甚至默认的 xmlns 也不是恒定的。

到目前为止,我提出的最佳工作解决方案是一组 regexp_replace(准确地说是 3 个),在解析之前擦除所有命名空间。但性能是一个巨大的问题。

当然,我缺少一些聪明的东西吗?

4

1 回答 1

5

我知道这已经很老了,但我今天发现了它,并记得我在尝试处理命名空间 XML 时所经历的痛苦。我的解决方案是使用 XSLT 转换去除名称空间,并将其作为普通的旧 XML 处理。我曾经这样做的功能是:

function remove_namespace( i_xml in xmltype )
  return xmltype
is
  v_xml xmltype default i_xml;
  v_xsl varchar2(32767);
begin
  v_xsl := '<?xml version="1.0" encoding="UTF-8"?>
        <xsl:stylesheet version="1.0"
         xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
        <xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
        <xsl:template match="*">
          <!-- remove element prefix (if any) -->
          <xsl:element name="{local-name()}">
          <!-- process attributes -->
          <xsl:for-each select="@*">
            <!-- remove attribute prefix (if any) -->
            <!-- this if filters out any xmlns="" atts that have no
                 namespace prefix in the xml -->
            <xsl:if test="(local-name() != ''xmlns'')">
              <xsl:attribute name="{local-name()}">
                <xsl:value-of select="."/>
              </xsl:attribute>
            </xsl:if>
          </xsl:for-each>
         <xsl:apply-templates/>
         </xsl:element>
         </xsl:template>
         </xsl:stylesheet>';
  return v_xml.transform(xmltype(v_xsl));
end;
于 2012-12-20T22:14:06.643 回答