gpt4 book ai didi

XSLT 1.0 : grouping and removing duplicate

转载 作者:行者123 更新时间:2023-12-01 07:58:39 24 4
gpt4 key购买 nike

我有一个 xml 分组挑战,我需要按如下方式分组并删除重复项:

<Person>
<name>John</name>
<date>June12</date>
<workTime taskID=1>34</workTime>
<workTime taskID=1>35</workTime>
<workTime taskID=2>12</workTime>
</Person>
<Person>
<name>John</name>
<date>June13</date>
<workTime taskID=1>21</workTime>
<workTime taskID=2>11</workTime>
<workTime taskID=2>14</workTime>
</Person>

请注意,对于 name/taskID/date 的特定出现,只会选取第一个。在这个例子中,

<workTime taskID=1>35</workTime> 
<workTime taskID=2>14</workTime>

会被放在一边。

下面是预期的输出:

<Person>
<name>John</name>
<taskID>1</taskID>
<workTime>
<date>June12</date>
<time>34</time>
</worTime>
<workTime>
<date>June13</date>
<time>21</time>
</worTime>
</Person>
<Person>
<name>John</name>
<taskID>2</taskID>
<workTime>
<date>June12</date>
<time>12</time>
</worTime>
<workTime>
<date>June13</date>
<time>11</time>
</worTime>
</Person>

我尝试使用以下键在 XSLT 1.0 中使用 muenchian 分组:

<xsl:key name="PersonTasks" match="workTime" use="concat(@taskID, ../name)"/>

但是我如何只获取第一次出现的

concat(@taskID, ../name, ../date)

?看来我需要两级 key !?

最佳答案

这个转换:

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

<xsl:key name="kwrkTimeByNameTask" match="workTime"
use="concat(../name, '+', @taskID)"/>

<xsl:key name="kDateByName" match="date"
use="../name"/>

<xsl:key name="kwrkTimeByNameTaskDate" match="workTime"
use="concat(../name, '+', @taskID, '+', ../date)"/>

<xsl:template match="/">
<xsl:for-each select=
"*/*/workTime
[generate-id()
=
generate-id(key('kwrkTimeByNameTask',
concat(../name, '+', @taskID)
)[1]
)
]
">
<xsl:sort select="../name"/>
<xsl:sort select="@taskID" data-type="number"/>

<xsl:variable name="vcurTaskId" select="@taskID"/>
<Person>
<name><xsl:value-of select="../name"/></name>
<taskID><xsl:value-of select="@taskID"/></taskID>

<xsl:for-each select=
"key('kDateByName', ../name)
[key('kwrkTimeByNameTaskDate',
concat(../name, '+', current()/@taskID, '+', .)
)
]
">
<workTime>
<date><xsl:value-of select="."/></date>
<time>
<xsl:value-of select=
"key('kwrkTimeByNameTaskDate',
concat(../name, '+', $vcurTaskId, '+', .)
)"/>
</time>
</workTime>
</xsl:for-each>
</Person>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>

应用于提供的 XML 时(从多个问题中更正为格式正确):

<t>
<Person>
<name>John</name>
<date>June12</date>
<workTime taskID="1">34</workTime>
<workTime taskID="1">35</workTime>
<workTime taskID="2">12</workTime>
</Person>
<Person>
<name>John</name>
<date>June13</date>
<workTime taskID="1">21</workTime>
<workTime taskID="2">11</workTime>
<workTime taskID="2">14</workTime>
</Person>
</t>

产生想要的、正确的结果:

<Person>
<name>John</name>
<taskID>1</taskID>
<workTime>
<date>June12</date>
<time>34</time>
</workTime>
<workTime>
<date>June13</date>
<time>21</time>
</workTime>
</Person>
<Person>
<name>John</name>
<taskID>2</taskID>
<workTime>
<date>June12</date>
<time>12</time>
</workTime>
<workTime>
<date>June13</date>
<time>11</time>
</workTime>
</Person>

解释:

  1. 首先我们获得所有workTime具有独特的 ../name 对的元素, @taskID 通过使用 Muenchian 方法进行分组。

  2. 我们根据 ../name 对这些进行排序@taskID -- 按此顺序。

  3. 对于每个这样的 workTime我们得到所有 date../name 一起列出的元素这个 workTime只留下这些 date元素,其中有一个 workTime具有相同的 ../date../name .

  4. 在上一步中我们使用了两个不同的辅助键:'kDateByName'索引所有date元素由他们 ../name , 而 'kwrkTimeByNameTaskDate'索引所有workTime元素由他们 ../name , 他们的 ../date和他们的 @taskID .

所以,下面的意思:

          <xsl:for-each select=
"key('kDateByName', ../name)
[key('kwrkTimeByNameTaskDate',
concat(../name, '+', current()/@taskID, '+', .)
)
]
">

是:

每个 date 为此 name , 这样一个 workTime 为此 name , date @taskID (当前的 workTime 外部 <xsl:for-each> )存在,做任何在这个主体中的事情 <xsl:for-each> 说明

关于XSLT 1.0 : grouping and removing duplicate,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3518399/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com