XSLT转换问题

时间:2016-05-06 18:16:25

标签: xml xslt

我尝试使用此源并使用以下规则对其进行转换:
1. Entry元素的数量会有所不同,因此它必须是for-each循环和 2.仅获取包含 CATDV

的条目元素
<?xml version="1.0"?>
<Metadata>
  <Entry>
    <Tag>CatDV 1</Tag>
    <Value>CCC</Value>
  </Entry>
  <Entry>
    <Tag>CatDV 2</Tag>
    <Value>DDD</Value>
  </Entry>
  <Entry>
    <Tag>Something Else</Tag>
    <Value>EEE</Value>
  </Entry>
  <Entry>
    <Tag>CatDV 3</Tag>
    <Value>FFF</Value>
  </Entry>
</Metadata>

我已经尝试了这一点,但我似乎无法缩小范围以获得具有CATDV标签的值。

  <catdv version="2.0">
    <remoteID>
      <xsl:choose>
        <xsl:when test="/Metadata/Entry[Tag = 'CATDV RemoteID']">
          <xsl:value-of select="/Metadata/Entry[Tag = 'CATDV RemoteID']/Value"/>
        </xsl:when>
      </xsl:choose>
    </remoteID>
    <metadata>
      <xsl:choose>
        <xsl:when test="Metadata/Entry/Tag = 'CATDV Animal'">
          <xsl:for-each select="Metadata/Entry">
            <field>
              <xsl:value-of select="/Metadata/Entry/Value"/>
            </field>
          </xsl:for-each>
        </xsl:when>
      </xsl:choose>
    </metadata>
  </catdv>

最终我想要这个结果。

<catdv version="2.0">
    <metadata>
        <fieldname>CATDV 1</fieldname>
        <fieldvalue>CCC</fieldvalue>
        <fieldname>CatDV 2</fieldname>
        <fieldvalue>>DDD</fieldvalue>
        <fieldname>CatDV 3</fieldname>
        <fieldvalue>FFF</fieldvalue>
    </metadata>
</catdv>

4 个答案:

答案 0 :(得分:2)

简单来说;

<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="/Metadata">
    <catdv version="2.0">
        <metadata>
            <xsl:for-each select="Entry[contains(Tag, 'CatDV')]">
            <fieldname>
                <xsl:value-of select="Tag" />
            </fieldname>
            <fieldvalue>
                <xsl:value-of select="Value" />
            </fieldvalue>
            </xsl:for-each>
        </metadata>
    </catdv>
</xsl:template>

</xsl:stylesheet>

答案 1 :(得分:1)

一个简单的XSLT-1.0解决方案就是:

<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
  <xsl:output method="xml" indent="yes" />
  <xsl:template match="/Metadata"> 
    <catdv version="2.0">
      <metadata>
        <xsl:apply-templates select="Entry[contains(Tag,'CatDV')]" />
      </metadata>
    </catdv>
  </xsl:template>
  <xsl:template match="Entry"> 
    <fieldname><xsl:value-of select="Tag" /></fieldname>
    <fieldvalue><xsl:value-of select="Value" /></fieldvalue>
  </xsl:template>
</xsl:stylesheet>

根据需要输出

<?xml version="1.0"?>
<catdv version="2.0">
  <metadata>
    <fieldname>CatDV 1</fieldname>
    <fieldvalue>CCC</fieldvalue>
    <fieldname>CatDV 2</fieldname>
    <fieldvalue>DDD</fieldvalue>
    <fieldname>CatDV 3</fieldname>
    <fieldvalue>FFF</fieldvalue>
  </metadata>
</catdv>

答案 2 :(得分:0)

<xsl:for-each select="Metadata/Entry">
  <xsl:if test="contains(Tag, 'CatDV')">
    <field>
      <xsl:value-of select="Value"/>
    </field>
  </xsl:if>
</xsl:for-each>

如果您想使比较不区分大小写,请查看here

答案 3 :(得分:0)

我会这样做

<?xml version="1.0" encoding="utf-8"?>

<xsl:transform version="1.0"  
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> 
<xsl:output method="xml" indent="yes"/>

<xsl:template match="/">
    <catdv version="2.0">
        <metadata>
            <xsl:apply-templates select="Metadata/Entry[contains(Tag,'CatDV')]"/>
        </metadata>
    </catdv>
</xsl:template>

<xsl:template match="Entry">
    <fieldname><xsl:value-of select="Tag"/></fieldname>
    <fieldvalue><xsl:value-of select="Tag"/></fieldvalue>

但可能有其他方式--Stefan