使用xslt删除根元素

时间:2013-01-15 12:15:18

标签: xslt

我有一个xml架构,其结构类似于

<Level>
<Level1>...data...</Level1>
<Level2>...data...</Level2>
.
.
.
</Level>

我想删除<Level>标记。在xslt的帮助下,我该怎么做呢。

3 个答案:

答案 0 :(得分:2)

任何“如何保持我的大部分XML相同,但调整它的一些内容”的标准答案是“使用身份模板,然后根据您想要更改的特定内容覆盖它”

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
  <!-- omit the <?xml?> line in the output, it won't be well-formed anyway -->
  <xsl:output method="xml" omit-xml-declaration="yes" />

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

  <xsl:template match="/*">
    <xsl:apply-templates select="node()" />
  </xsl:template>
</xsl:stylesheet>

但正如Lister先生在评论中指出的那样,这将为您留下一些格式不正确的XML,因为它将包含多个文档元素。

当我在输入XML

上应用此样式表时
<Level>
<Level1>...data...</Level1>
<Level2>...data...</Level2>
</Level>

它产生结果

<Level1>...data...</Level1>
<Level2>...data...</Level2>

答案 1 :(得分:0)

如果要将文档元素的所有子元素存储在变量中,可以执行以下操作:

<xsl:variable name="myVar" select="/*/*"/>

但是,如果您希望样式表生成字符串,这可能是一个解决方案:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  version="1.0">
  <xsl:output method="text"/>

  <xsl:template match="/*">
    <xsl:apply-templates select="node()"/>
  </xsl:template>

  <xsl:template match="*">
    <!-- We write the opening tag -->
    <xsl:value-of select="concat('&lt;',local-name())"/>
    <!-- Now, all attributes -->
    <xsl:apply-templates select="@*"/>
    <!-- Depending on whether we have an empty element or not, 
         we're adding the content or closing it immediately -->
    <xsl:choose>
      <xsl:when test="node()">
        <!-- Close opening tag -->
        <xsl:value-of select="'&gt;'"/>
        <!-- Add the content -->
        <xsl:apply-templates select="node()"/>
        <!-- Write closing tag -->
        <xsl:value-of select="concat('&lt;/',local-name(),'&gt;')"/>
      </xsl:when>
      <xsl:otherwise>
        <!-- Create empty element by closing tag immediately -->
        <xsl:value-of select="'/&gt;'"/>
      </xsl:otherwise>
    </xsl:choose>
  </xsl:template>

  <xsl:template match="@*">
    <!-- Write an attribute -->
    <xsl:value-of select="concat(' ',local-name(),'=&quot;',.,'&quot;')"/>
  </xsl:template>
</xsl:stylesheet>

它生成文本(因此您不会获得格式不正确的XML)。它有点过于简化,因为它不处理属性中的命名空间,注释,处理指令和引号。如果您的输入XML包含其中任何一个,则必须优化样式表。

答案 2 :(得分:0)

我创建了一个新的XSLT定义,在您的代码的帮助下完成了我的请求

<?xml version="1.0" encoding="UTF-8"?><xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output encoding="utf-8" method="text" omit-xml-declaration="yes"/>
<xsl:variable name="nl">
    <xsl:text/>
</xsl:variable>
<xsl:variable name="tb">
    <xsl:text/>
</xsl:variable>
<xsl:template match="/*">
    <!-- Open the root array -->
    <xsl:text>[{</xsl:text>
    <xsl:value-of select="$nl"/>
    <!-- Process all the child nodes of the root -->
    <xsl:apply-templates mode="detect" select="*">
        <xsl:with-param name="indent" select="$tb"/>
    </xsl:apply-templates>
    <!-- Close the root array -->
    <xsl:value-of select="$nl"/>
    <xsl:text>}]</xsl:text>
</xsl:template>
<xsl:template match="*" mode="detect">
    <xsl:choose>
        <xsl:when test="name(preceding-sibling::*[1]) = name(current()) and name(following-sibling::*[1]) != name(current())">
            <xsl:apply-templates mode="obj-content" select="."/>
            <xsl:text>]</xsl:text>
            <xsl:if test="count(following-sibling::*[name() != name(current())]) > 0">, </xsl:if>
        </xsl:when>
        <xsl:when test="name(preceding-sibling::*[1]) = name(current())">
            <xsl:apply-templates mode="obj-content" select="."/>
            <xsl:if test="name(following-sibling::*) = name(current())">, </xsl:if>
        </xsl:when>
        <xsl:when test="following-sibling::*[1][name() = name(current())]">
            <xsl:text>"</xsl:text>
            <xsl:value-of select="name()"/>
            <xsl:text>" : [</xsl:text>
            <xsl:apply-templates mode="obj-content" select="."/>
            <xsl:text>, </xsl:text>
        </xsl:when>
        <xsl:when test="count(./child::*) > 0 or count(@*) > 0">
            <xsl:text>"</xsl:text>
            <xsl:value-of select="name()"/>" : [<xsl:apply-templates
                mode="obj-content" select="."/>
            <xsl:if test="count(following-sibling::*) > 0">], </xsl:if>
        </xsl:when>
        <xsl:when test="count(./child::*) = 0">
            <xsl:text>"</xsl:text>
            <xsl:value-of select="name()"/>" : "<xsl:apply-templates select="."/>
            <xsl:text>"</xsl:text>
            <xsl:if test="count(following-sibling::*) > 0">, </xsl:if>
        </xsl:when>
    </xsl:choose>
</xsl:template>
<xsl:template match="*" mode="obj-content">
    <xsl:text>{</xsl:text>
    <xsl:apply-templates mode="attr" select="@*"/>
    <xsl:if test="count(@*) > 0 and (count(child::*) > 0 or text())">, </xsl:if>
    <xsl:apply-templates mode="detect" select="./*"/>
    <xsl:if test="count(child::*) = 0 and text() and not(@*)">
        <xsl:text>"</xsl:text>
        <xsl:value-of select="name()"/>" : "<xsl:value-of select="text()"/>
        <xsl:text>"</xsl:text>
    </xsl:if>
    <xsl:if test="count(child::*) = 0 and text() and @*">
        <xsl:text>: "</xsl:text>
        <xsl:value-of select="text()"/>
        <xsl:text>"</xsl:text>
    </xsl:if>
    <xsl:text>}</xsl:text>
    <xsl:if test="position() &lt; last()">, </xsl:if>
</xsl:template>
<xsl:template match="@*" mode="attr">
    <xsl:text>"</xsl:text>
    <xsl:value-of select="name()"/>" : "<xsl:value-of select="."/>
    <xsl:text>"</xsl:text>
    <xsl:if test="position() &lt; last()">,</xsl:if>
</xsl:template>
<xsl:template match="node/@TEXT | text()" name="removeBreaks">
    <xsl:param name="pText" select="normalize-space(.)"/>
    <xsl:choose>
        <xsl:when test="not(contains($pText, '&#xa;'))">
            <xsl:copy-of select="$pText"/>
        </xsl:when>
        <xsl:otherwise>
            <xsl:value-of select="concat(substring-before($pText, '&#xd;&#xa;'), ' ')"/>
            <xsl:call-template name="removeBreaks">
                <xsl:with-param name="pText" select="substring-after($pText, '&#xd;&#xa;')"/>
            </xsl:call-template>
        </xsl:otherwise>
    </xsl:choose>
</xsl:template>