XSLT删除空白名称空间而不将元素分配给不同的名称空间或默认名称空间

时间:2015-07-15 21:01:48

标签: xml xslt namespaces

我有以下XML:

<ns1:A xmlns:ns1="http://www.namespace1.com" xmlns:ns2="http://www.namespace2.com">
<B>
    <C>123</C>
</B>
<ns2:D>
    <E>456</E>
</ns2:D>
</ns1:A>

我想使用以下XSLT对其进行转换:

<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
            xmlns:xs="http://www.w3.org/2001/XMLSchema" 
            xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
            exclude-result-prefixes="xsl xs xsi">

<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>

<xsl:strip-space elements="*" />

<xsl:template match="/">
    <xsl:apply-templates />
</xsl:template>

<xsl:template match="*">
    <xsl:element name="{local-name()}" namespace="{namespace-uri()}">
        <xsl:copy-of select="@*|namespace::*[name()]"/>
        <xsl:if test="text() != ''">
            <xsl:value-of select="concat(text(), 'xyz')"/>
        </xsl:if>
        <xsl:apply-templates select="*"/>
    </xsl:element>
</xsl:template>
</xsl:stylesheet>

输出具有元素B和E的不需要的空白名称空间:

<A xmlns="http://www.namespace1.com" xmlns:ns1="http://www.namespace1.com" xmlns:ns2="http://www.namespace2.com">
<B xmlns="">
  <C>123xyz</C>
</B>
<D xmlns="http://www.namespace2.com">
  <E xmlns="">456xyz</E>
</D>
</A>

我希望删除输出中的空白名称空间。我见过很多关于分配默认命名空间或将元素分配给某个父命名空间的帖子。然而,这不是我想要的。在输出中,我确实希望元素B,C和E保留在“空白”命名空间中,因为这是相应的XSD所需要的(并且由于遗留原因我无法更新XSD)。所以期望的输出是:

 <A xmlns="http://www.namespace1.com" xmlns:ns1="http://www.namespace1.com" xmlns:ns2="http://www.namespace2.com">
 <B>
   <C>123xyz</C>
 </B>
 <D xmlns="http://www.namespace2.com">
   <E>456xyz</E>
 </D>
</A>

如何摆脱输出中的空白名称空间?我还想指出,XSLT将应用于具有不同模式结构的不同输入XML,因此XSLT必须尽可能通用,而无需对元素或命名空间名称进行硬编码。

编辑:

回顾M. Kay和M. Hor的答案,我意识到我以前的“所需”输出是不正确的。正如M. Hor所解释的那样,默认名称空间是继承的,如我之前的“所需”输出中所列,这也不是我想要的。 (抱歉输入错误的“所需”)。

我的(更正的)所需输出是不使用任何默认命名空间,而是使用命名空间前缀:

<ns1:A xmlns:ns1="http://www.namespace1.com" xmlns:ns2="http://www.namespace2.com">
   <B>
      <C>123xyz</C>
   </B>
   <ns2:D>
      <E>456xyz</E>
   </ns2:D>
</ns1:A>

我已经更新了我的XSLT作为下面这个问题的答案,它输出了我想要的内容。

谢谢! IK

2 个答案:

答案 0 :(得分:1)

  

(输入和输出是相同的,因为我使用简化   XSLT现在。)

当您的XSLT实际上没有改变任何东西时,很难理解这个问题。对于声明的目的,identity transform模板应该可以正常工作:

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="2.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<!-- identity transform -->
<xsl:template match="@*|node()">
    <xsl:copy>
        <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:template>

</xsl:stylesheet>

修改

响应您的修改,显示不同的所需输出:

你这么说:

  

我确实希望元素B,C和E保留在&#34;空白&#34;命名空间

您所需的输出中会发生什么:

<A xmlns="http://www.namespace1.com" xmlns:ns1="http://www.namespace1.com"
   xmlns:ns2="http://www.namespace2.com">
   <B>
      <C>123xyz</C>
   </B>
   <D xmlns="http://www.namespace2.com">
      <E>456xyz</E>
   </D>
</A>

此处,元素BC从其A祖先继承其命名空间。同样,E元素与其D父元素位于同一名称空间中。

当您声明默认命名空间时,它是如何工作的:声明范围内的所有元素都放在该命名空间中。为了将B置于无命名空间中,您必须进行显式异常。这就是您在实际结果中看到<B xmlns="">的原因。

答案 1 :(得分:0)

以下XSLT将输出名称空间前缀而不输出默认名称空间,从而避免输出中的xmlns =“”。

<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" exclude-result-prefixes="xsl xs xsi">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:template match="/">
    <xsl:apply-templates/>
</xsl:template>

<xsl:template match="*">
    <xsl:element name="{name()}" namespace="{namespace-uri()}">
        <xsl:copy-of select="@*|namespace::*[name()]"/>
        <xsl:if test="text() != ''">
            <xsl:value-of select="concat(text(), 'xyz')"/>
        </xsl:if>
        <xsl:apply-templates select="*"/>
    </xsl:element>
</xsl:template>

</xsl:stylesheet>

基本上,我没有使用local-name(),而是使用name()来包含前缀。

输出现在是:

<ns1:A xmlns:ns1="http://www.namespace1.com" xmlns:ns2="http://www.namespace2.com">
<B>
  <C>123xyz</C>
</B>
<ns2:D>
  <E>456xyz</E>
</ns2:D>
</ns1:A>

注意:这是使用Saxon Home Edition 9.4测试的。我注意到其他XSLT处理器可能仍然输出xmlns =“”。