使用XSLT创建表,其值为与表头相关的记录

时间:2016-07-13 12:28:24

标签: xml xslt distinct

我有XML

     <People>
        <Person>
          <Index>0</Index>
          <Attributes>
            <Attribute>
              <Name>First Name</Name>
              <Value>Mike</Value>
            </Attribute>
            <Attribute>
              <Name>Country</Name>
              <Value>France</Value>
            </Attribute>
            <Attribute>
              <Name>City</Name>
              <Value>Paris</Value>
            </Attribute>
          </Attributes>
        </Person>
        <Person>
          <Index>1</Index>
          <Attributes>
            <Attribute>
              <Name>First Name</Name>
              <Value>Peter</Value>
            </Attribute>
            <Attribute>
              <Name>Country</Name>
              <Value>Germany</Value>
            </Attribute>
            <Attribute>
              <Name>Height</Name>
              <Value>190</Value>
            </Attribute>
          </Attributes>
        </Person>
        <Person>
          <Index>2</Index>
          <Attributes>
            <Attribute>
              <Name>First Name</Name>
              <Value>Justine</Value>
            </Attribute>
            <Attribute>
              <Name>Hobby</Name>
              <Value>Volleyball</Value>
            </Attribute>
          </Attributes>
        </Person>
      </People>

我的输出应该是

First Name  Country City    Height  Hobby
Mike        France  Paris       
Peter       Germany         190 
Justine                             Volleyball

表的标题具有不同的值 但如何将记录单元格与列相关联?

我试图使用generate-id()和distinct-values()但是值不在好列上

我的代码是

<xsl:key name="KeyName" match="People/Person/Attributes/Attribute" use="Name" />
<xsl:template match="People">
<table>
        <tr>
            <xsl:for-each select="Person/Attributes/Attribute[generate-id() = generate-id(key('KeyName', Name)[1])]">
                <th>    <xsl:value-of select="Name"/> </th>
            </xsl:for-each>
        </tr>

    <xsl:for-each select="Person">
        <tr>
            <xsl:for-each select="Attributes/Attribute">            
                <td>    <xsl:value-of select="Value"/>  </td>
            </xsl:for-each>
        </tr>
    </xsl:for-each>
</table>
</xsl:template>

我的错误代码可能暗示某人错误地走了。

我的主张是 使用generate-id在标头中包含不同的值并将每个插入到(th) 记得这个清单。

将每个记录排序到此标题列表,比较标记名称 如果比较标记Name在Person的记录中不存在,我们插入空值“”

现在在Person中排序值(标记值与标题相关),我们在表中插入行。

我使用XSLT 2.0,但版本没问题

有什么想法吗?

1 个答案:

答案 0 :(得分:0)

XSLT 2.0 中,您可以这样做:

<xsl:stylesheet version="2.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" omit-xml-declaration="yes" version="1.0" encoding="utf-8" indent="yes"/>

<xsl:key name="attr-by-name" match="Attribute" use="Name" />

<xsl:template match="/People">
    <xsl:variable name="attributes" select="distinct-values(Person/Attributes/Attribute/Name)" />
    <table border="1">
        <tr>
            <xsl:for-each select="$attributes">
                <th>
                    <xsl:value-of select="."/>
                </th>
            </xsl:for-each>
        </tr>
        <xsl:for-each select="Person">
            <xsl:variable name="my-attributes" select="Attributes" />
            <tr>
                <xsl:for-each select="$attributes">            
                    <td>
                        <xsl:value-of select="key('attr-by-name', ., $my-attributes)/Value"/>
                    </td>
                </xsl:for-each>
            </tr>
        </xsl:for-each>
    </table>
</xsl:template>

</xsl:stylesheet>

应用于您的输入示例,(呈现)结果将为:

enter image description here