使用 XSL 将 Sparsely-Populated XML 列表转换为 HTML Table
Convert Sparsely-Populated XML List to HTML Table Using XSL
我正在尝试使用 XSL 将 XML 文件转换为 nice-looking HTML table,但我终究无法弄清楚如何正确排列所有列。我发现了一个类似的问题 (),它提供了很多信息(一般来说我对 XSL 还很陌生,所以一点点帮助),但是我的 XML 的结构有点……不规则:
<?xml version="1.0"?>
<root>
<Group>
<Type>Type1</Type>
<Desc>Desc1</Desc>
<Name>Name1</Name>
</Group>
<Group>
<Type>Type2</Type>
<Name>Name2</Name>
<State>State2</State>
</Group>
<Group>
<Type>Type3</Type>
<State>State3</State>
<Country>Country4</Country>
</Group>
</root>
有什么好的方法可以把它变成well-formattedtable吗?我更希望它能像这样在 XML 上工作,就好像我们曾经向其中一个 Group 元素添加一个新字段一样,我不希望他们必须向每个元素添加相同的标签其他元素(尽管如果有交替的好方法自动将所有缺失的children添加到其他组元素中存在的组元素,那也很好,也许使用可以 运行 首先的第二个 XSLT)。此 XML 没有支持模式,但如果有帮助,可以制作一个。
抱歉,如果之前已经回答过这个问题,除了提到的问题之外,我找不到更接近的问题,也没有关于如何将缺失的 children 添加到元素的信息。
首先,这并不简单。如果您是 XSLT 的新手,那么这可能不是最适合您的项目。
为了理解以下解决方案,您需要熟悉使用 keys, as well as the Muenchian grouping 技术。
XSLT 1.0
<xsl:stylesheet version="1.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="col-by-name" match="/root/Group/*" use="name()" />
<xsl:key name="cell-by-rowXcol" match="/root/Group/*" use="concat(generate-id(..), '|', name())" />
<xsl:template match="/root">
<xsl:variable name="distinct-columns" select="Group/*[count(. | key('col-by-name', name())[1]) = 1]"/>
<table border="1">
<thead>
<tr>
<xsl:for-each select="$distinct-columns">
<th><xsl:value-of select="name()"/></th>
</xsl:for-each>
</tr>
</thead>
<tbody>
<xsl:for-each select="Group">
<xsl:variable name="row" select="generate-id()" />
<tr>
<xsl:for-each select="$distinct-columns">
<td>
<xsl:value-of select="key('cell-by-rowXcol', concat($row, '|', name()))" />
</td>
</xsl:for-each>
</tr>
</xsl:for-each>
</tbody>
</table>
</xsl:template>
</xsl:stylesheet>
结果(渲染):
在 XSLT 2.0 中,您可以使用 distinct-values()
函数代替 Muenchian 分组。
这是我对 XSLT 2.0 解决方案的尝试:
<xsl:transform xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="2.0">
<xsl:output method="html" version="5.0" encoding="UTF-8" indent="yes" />
<xsl:variable name="col-names" select="distinct-values(root/Group/*/node-name(.))"/>
<xsl:template match="root">
<table>
<thead>
<tr>
<xsl:for-each select="$col-names">
<th>
<xsl:value-of select="."/>
</th>
</xsl:for-each>
</tr>
</thead>
<tbody>
<xsl:apply-templates select="Group"/>
</tbody>
</table>
</xsl:template>
<xsl:template match="Group">
<tr>
<xsl:variable name="this" select="."/>
<xsl:for-each select="$col-names">
<td>
<xsl:value-of select="$this/*[node-name(.) eq current()]"/>
</td>
</xsl:for-each>
</tr>
</xsl:template>
</xsl:transform>
我正在尝试使用 XSL 将 XML 文件转换为 nice-looking HTML table,但我终究无法弄清楚如何正确排列所有列。我发现了一个类似的问题 (
<?xml version="1.0"?>
<root>
<Group>
<Type>Type1</Type>
<Desc>Desc1</Desc>
<Name>Name1</Name>
</Group>
<Group>
<Type>Type2</Type>
<Name>Name2</Name>
<State>State2</State>
</Group>
<Group>
<Type>Type3</Type>
<State>State3</State>
<Country>Country4</Country>
</Group>
</root>
有什么好的方法可以把它变成well-formattedtable吗?我更希望它能像这样在 XML 上工作,就好像我们曾经向其中一个 Group 元素添加一个新字段一样,我不希望他们必须向每个元素添加相同的标签其他元素(尽管如果有交替的好方法自动将所有缺失的children添加到其他组元素中存在的组元素,那也很好,也许使用可以 运行 首先的第二个 XSLT)。此 XML 没有支持模式,但如果有帮助,可以制作一个。
抱歉,如果之前已经回答过这个问题,除了提到的问题之外,我找不到更接近的问题,也没有关于如何将缺失的 children 添加到元素的信息。
首先,这并不简单。如果您是 XSLT 的新手,那么这可能不是最适合您的项目。
为了理解以下解决方案,您需要熟悉使用 keys, as well as the Muenchian grouping 技术。
XSLT 1.0
<xsl:stylesheet version="1.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="col-by-name" match="/root/Group/*" use="name()" />
<xsl:key name="cell-by-rowXcol" match="/root/Group/*" use="concat(generate-id(..), '|', name())" />
<xsl:template match="/root">
<xsl:variable name="distinct-columns" select="Group/*[count(. | key('col-by-name', name())[1]) = 1]"/>
<table border="1">
<thead>
<tr>
<xsl:for-each select="$distinct-columns">
<th><xsl:value-of select="name()"/></th>
</xsl:for-each>
</tr>
</thead>
<tbody>
<xsl:for-each select="Group">
<xsl:variable name="row" select="generate-id()" />
<tr>
<xsl:for-each select="$distinct-columns">
<td>
<xsl:value-of select="key('cell-by-rowXcol', concat($row, '|', name()))" />
</td>
</xsl:for-each>
</tr>
</xsl:for-each>
</tbody>
</table>
</xsl:template>
</xsl:stylesheet>
结果(渲染):
在 XSLT 2.0 中,您可以使用 distinct-values()
函数代替 Muenchian 分组。
这是我对 XSLT 2.0 解决方案的尝试:
<xsl:transform xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="2.0">
<xsl:output method="html" version="5.0" encoding="UTF-8" indent="yes" />
<xsl:variable name="col-names" select="distinct-values(root/Group/*/node-name(.))"/>
<xsl:template match="root">
<table>
<thead>
<tr>
<xsl:for-each select="$col-names">
<th>
<xsl:value-of select="."/>
</th>
</xsl:for-each>
</tr>
</thead>
<tbody>
<xsl:apply-templates select="Group"/>
</tbody>
</table>
</xsl:template>
<xsl:template match="Group">
<tr>
<xsl:variable name="this" select="."/>
<xsl:for-each select="$col-names">
<td>
<xsl:value-of select="$this/*[node-name(.) eq current()]"/>
</td>
</xsl:for-each>
</tr>
</xsl:template>
</xsl:transform>