我有两个xmls,User.xml和UIDL Rules.xml
user.xml 就是这样..
<UI-defination>
<class name="User">
<list_view>
<members>
<member name="code"/>
<member name="creationTS"/>
<member name="creator"/>
<member name="displayName"/>
<member name="emailAddress"/>
<member name="id"/>
<member name="loginId"/>
<member name="password"/>
<member name="remarks"/>
<member name="status"/>
</members>
</list_view>
</class>
</UI-defination>
我的 UIDL Rules.xml 包含
<?xml version="1.0" encoding="UTF-8"?>
<UI-defination>
<class name="Role">
<list_view multiselect="true">
<members>
<member name="loginId" sequence="4" />
<member name="code" sequence="1" />
<member name="status" sequence="6"/>
</members>
<exclude>
<members>
<member name="id"/>
<member name="creator"/>
</members>
</exclude>
</list_view>
</class>
</UI-defination>
我的xslt代码
<xsl:stylesheet version="2.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:xs="http://www.w3.org 2001/XMLSchema"
xmlns:functx="http://www.functx.com">
<xsl:variable name="uiRules"
select="document('UIDL Rules.xml')" />
<xsl:variable name="reserved-members">
<xsl:for-each
select="$uiRules//UI-defination/class/list_view/members/member[@sequence]">
<xsl:sort select="@sequence" data-type="number" order="ascending" />
<xsl:copy-of select="." />
</xsl:for-each>
</xsl:variable>
<xsl:key name="member" match="list_view/members/member" use="@name" />
<xsl:variable name="exclude"
select="key('member', $uiRules//exclude/members/member/@name)" />
<xsl:variable name="exclude1"
select="key('member', $uiRules//UI-defination/class/list_view/members/member[@sequence]/@name)" />
<xsl:variable name="ordinary-members"
select="//list_view/members/member except $exclude" />
<xsl:variable name="ordinary-members1" select="$ordinary-members except $exclude1" />
<xsl:template match="/">
<xsl:for-each select="$reserved-members/member">
<xsl:variable name="previous-place">
<xsl:choose>
<xsl:when test="position()>1">
<xsl:value-of select="preceding-sibling::member[1]/@sequence" />
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="0" />
</xsl:otherwise>
</xsl:choose>
</xsl:variable>
<xsl:variable name="gap-size" select="@sequence - $previous-place - 1"/>
<xsl:variable name="gap-start" select="$previous-place - count(preceding-sibling::member) + 1"/>
<xsl:for-each select="$ordinary-members1[$gap-start <= position() and position() < $gap-start + $gap-size]">
<xsl:variable name="currentMember" select="@name" />
<xsl:value-of select="$currentMember" />
</xsl:for-each>
<xsl:variable name="currentMember" select="@name" />
<xsl:value-of select="$currentMember" />
<!-- output remaining ordinary members -->
<xsl:if test="position()=last()">
<xsl:for-each
select="$ordinary-members1[position() >= $gap-start + $gap-size]">
<xsl:variable name="currentMember" select="@name" />
<xsl:value-of select="$currentMember"/>
</xsl:for-each>
</xsl:if>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
我的要求是从 user.xml 列出成员减去 UIDL Rule.xml 中的排除成员,并根据序列排列其余成员属性。上面的代码工作正常但我必须在15个不同的地方使用这个逻辑,所以如果我将输出放在一个变量中,其值将由逗号分隔的成员,那将会很棒。 有什么建议吗?
Desired output sequence should look like this
1>code
2>creationTS
3>displayName
4>loginId
5>emailAddress
6>status
7>password
8>remarks
答案 0 :(得分:1)
我知道这个问题已经过了一个星期,并没有得到OP的重新关注,但无论如何我都会试一试。
我尝试了你的代码,但正如Martin Honnen已经指出的那样,它已被打破。标签未正确关闭且变量丢失。此外,它包含了帖子中未涵盖的更多信息,因此即使在我添加变量并尝试使用之后,它也没有给出您在帖子中描述的结果。
所以,因为要求是如此基本,我想,让我们试一试。我理解的要求是:
- 从user.xml
获取所有成员
- 删除exclude
中UIDL Rule.xml
部分中具有相同名称的成员(这不是有效的URI,因此我删除了空格)。
- 根据规则文件中相应sequence
属性中的值对成员进行排序。
这就是我想出来的,它完全符合您的要求:
<xsl:stylesheet version="2.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xsl:output indent="yes" />
<xsl:variable name="uiRules" select="document('UIDLRules.xml')" />
<xsl:variable name="excl" select="$uiRules/*/*/*/exclude/members/member/data(@name)" />
<xsl:template match="/">
<xsl:variable name="unsorted" as="element()*">
<xsl:apply-templates />
</xsl:variable>
<xsl:for-each select="$unsorted">
<xsl:sort select="@sequence" order="descending" />
<xsl:copy-of select="." />
</xsl:for-each>
</xsl:template>
<xsl:template match="text()" />
<xsl:template match="member[not(@name = $excl)]">
<xsl:copy>
<xsl:apply-templates select="@name" />
</xsl:copy>
</xsl:template>
<xsl:template match="@name">
<xsl:copy />
<xsl:copy-of select="$uiRules/*/*/*/members/member[@name = current()]/@sequence" />
</xsl:template>
</xsl:stylesheet>
使用您的输入,这是输出(即,id
和creator
被过滤掉,顺序基于sequence
属性。
<?xml version="1.0" encoding="UTF-8"?>
<member name="status" sequence="6"/>
<member name="loginId" sequence="4"/>
<member name="code" sequence="1"/>
<member name="creationTS"/>
<member name="displayName"/>
<member name="emailAddress"/>
<member name="password"/>
<member name="remarks"/>
我使用的技巧很简单:只需使用rules xml文件中的额外属性扩充原始输入member
元素,然后进行排序。
你的问题是如何在一个变量中捕获它的(好吧,你说的是“ this for-each循环,但你有多个它们我只有我的解决方案中的一个)。
此外,由于您的代码中已经有这么多变量,我不确定我是否正确理解您的问题。但是要切入追逐,您可以将xsl:for-each
包装在变量中,这将允许您在本地级别重用,或者您可以将整个处理包装在变量中。
即,您可以这样做来创建一个包含已排序成员列表的全局变量:
<!-- global variable for re-use -->
<xsl:variable name="sorted">
<xsl:variable name="unsorted" as="element()*">
<xsl:apply-templates />
</xsl:variable>
<xsl:for-each select="$unsorted">
<xsl:sort select="@sequence" order="descending" />
<xsl:copy-of select="." />
</xsl:for-each>
</xsl:variable>
<xsl:template match="/">
<!-- the central match is now replaced by a simple copy -->
<xsl:copy-of select="$sorted" />
</xsl:template>
这是有效的,因为全局上下文项与初始模式的初始上下文项相同。换句话说,您可以通过仅应用模板来引用原始输入文档。在上述更改之后,输出未更改,但您可以根据需要重复使用该变量。
请注意,已知的XSLT 2.0处理器,例如Exselt(免责声明,我是此开发人员),Saxon或来自Altova缓存全局变量的处理器,因此整个处理该变量只会发生一次。