我正试着围绕xslt。 关于stackoverflow的一些问题有帮助( XSLT templates and recursion 和 XSLT for-each loop, filter based on variable )但我仍然有点困惑。 我想我“把模板当作功能” (https://stackoverflow.com/questions/506348/how-do-i-know-my-xsl-is-efficient-and-beautiful)
无论如何......我的数据是
<Entities>
<Entity ID="8" SortValue="0" Name="test" ParentID="0" />
<Entity ID="14" SortValue="2" Name="test2" ParentID="8" />
<Entity ID="16" SortValue="1" Name="test3" ParentID="8" />
<Entity ID="17" SortValue="3" Name="test4" ParentID="14" />
<Entity ID="18" SortValue="3" Name="test5" ParentID="0" />
</Entities>
我想要的输出基本上是“树视图”
<ul>
<li id="entity8">
test
<ul>
<li id="entity16">
test3
</li>
<li id="entity14">
test2
<ul>
<li id="entity17">
test4
</li>
</ul>
</li>
</ul>
</li>
<li id="entity18">
test5
</li>
</ul>
我到目前为止的XSLT错误之处在于它确实“将模板视为函数”并且在执行时抛出了一个StackOverflowException(:-))
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:msxsl="urn:schemas-microsoft-com:xslt" exclude-result-prefixes="msxsl">
<xsl:output method="html" indent="yes"/>
<xsl:template match="Entities">
<ul>
<li>
<xsl:value-of select="local-name()"/>
<xsl:apply-templates/>
</li>
</ul>
</xsl:template>
<xsl:template match="//Entities/Entity[@ParentID=0]">
<xsl:call-template name="recursive">
<xsl:with-param name="parentEntityID" select="0"></xsl:with-param>
</xsl:call-template>
</xsl:template>
<xsl:template name="recursive">
<xsl:param name="parentEntityID"></xsl:param>
<xsl:variable name="counter" select="//Entities/Entity[@ParentID=$parentEntityID]"></xsl:variable>
<xsl:if test="count($counter) > 0">
<xsl:if test="$parentEntityID > 0">
</xsl:if>
<li>
<xsl:variable name="entityID" select="@ID"></xsl:variable>
<xsl:variable name="sortValue" select="@SortValue"></xsl:variable>
<xsl:variable name="name" select="@Name"></xsl:variable>
<xsl:variable name="parentID" select="@ParentID"></xsl:variable>
<a href=?ID={$entityID}&ParentEntityID={$parentID}">
<xsl:value-of select="$name"/>
</a>
<xsl:call-template name="recursive">
<xsl:with-param name="parentEntityID" select="$entityID"></xsl:with-param>
</xsl:call-template>
</li>
</xsl:if>
</xsl:template>
</xsl:stylesheet>
我知道如何通过代码执行此操作,没问题。 不过,这一次,我正在寻找xslt的解决方案,并且非常感谢任何帮助。
答案 0 :(得分:6)
虽然call-template
和命名模板是该语言的一个非常有用的功能,但如果您发现自己更喜欢apply-templates
,则可能表明您仍在考虑功能而非模板。如果您在命名模板中执行的第一件事是选择要操作的节点集,则尤其如此。
以下是您尝试执行的操作的简单版本。
<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="/">
<ul>
<xsl:apply-templates select="Entities/Entity[@ParentID=0]" />
</ul>
</xsl:template>
<xsl:template match="Entity">
<li>
<xsl:value-of select="@Name" />
<xsl:apply-templates select="../Entity[@ParentID=current()/@ID]" />
</li>
</xsl:template>
</xsl:stylesheet>
请注意,不需要计数器,因为“父”的值提供了必要的上下文。
另请注意,所有Entities
行为方式相同,无论它们在树中的位置如何,它们都包含@Name
值,并将模板应用于任何Entity
个对象, @ParentID匹配当前级别的@ID。
答案 1 :(得分:1)
正确和高效的解决方案(当前接受的答案不会产生有用的嵌套ul
元素:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:key name="kChildren" match="Entity" use="@ParentID"/>
<xsl:template match="/*[Entity]">
<ul>
<xsl:apply-templates select="key('kChildren', '0')">
<xsl:sort select="@SortValue" data-type="number"/>
</xsl:apply-templates>
</ul>
</xsl:template>
<xsl:template match="Entity">
<li id="entity{@ID}">
<xsl:value-of select="concat('
 ', @Name, '
')"/>
<xsl:if test="key('kChildren', @ID)">
<ul>
<xsl:apply-templates select="key('kChildren', @ID)">
<xsl:sort select="@SortValue" data-type="number"/>
</xsl:apply-templates>
</ul>
</xsl:if>
</li>
</xsl:template>
</xsl:stylesheet>
在提供的XML文档上应用此转换时:
<Entities>
<Entity ID="8" SortValue="0" Name="test" ParentID="0" />
<Entity ID="14" SortValue="2" Name="test2" ParentID="8" />
<Entity ID="16" SortValue="1" Name="test3" ParentID="8" />
<Entity ID="17" SortValue="3" Name="test4" ParentID="14" />
<Entity ID="18" SortValue="3" Name="test5" ParentID="0" />
</Entities>
产生了想要的正确结果:
<ul>
<li id="entity8">
test
<ul>
<li id="entity16">
test3
</li>
<li id="entity14">
test2
<ul>
<li id="entity17">
test4
</li>
</ul>
</li>
</ul>
</li>
<li id="entity18">
test5
</li>
</ul>