XSLT用于包装特定条件下的紧随的后续同级元素

huangapple go评论77阅读模式
英文:

XSLT for wrapping immediate following-siblings of certain condition

问题

以下是您提供的XML的翻译,包括XSLT代码部分:

  1. <root>
  2. <i type="r"/>
  3. <i type="s"/>
  4. <i type="r"/>
  5. <i type="r"/>
  6. <i type="s"/>
  7. <i type="r"/>
  8. <i type="s"/>
  9. </root>

我的目标是将所有 <i type="s"> 节点与所有紧随其后的类型为 "r" 的节点(1..n)一起包装起来。

不包装的情况:

  • 没有前置 "s" 节点的 "r" 节点
  • 没有紧随后续 "r" 节点的 "s" 节点

预期输出:

  1. <i type="r"/>
  2. <wrap>
  3. <i type="s"/>
  4. <i type="r"/>
  5. <i type="r"/>
  6. </wrap&gt;
  7. &lt;wrap&gt;
  8. &lt;i type=&quot;s&quot;/&gt;
  9. &lt;i type=&quot;r&quot;/&gt;
  10. </wrap&gt;
  11. &lt;i type=&quot;s&quot;/&gt;

我一直在尝试使用以下XSLT代码片段解决问题:

  1. &lt;xsl:for-each select=&quot;./i&quot;&gt;
  2. &lt;xsl:choose&gt;
  3. &lt;xsl:when test=&quot;current()[./@type=&#39;r&#39; and count(preceding-sibling::i[@type=&#39;s&#39;]) = 0]&quot;&gt;
  4. &lt;!-- 处理不包装 --&gt;
  5. &lt;xsl:apply-templates select=&quot;current()&quot; mode=&quot;fill&quot;/&gt;
  6. &lt;/xsl:when&gt;
  7. &lt;xsl:when test=&quot;current()[./@type=&#39;s&#39; and following-sibling::i[@type=&#39;s&#39;]]&quot;&gt;
  8. &lt;!-- 处理不包装 --&gt;
  9. &lt;xsl:apply-templates select=&quot;current()&quot; mode=&quot;fill&quot;/&gt;
  10. &lt;/xsl:when&gt;
  11. &lt;xsl:when test=&quot;current()[./@type=&#39;r&#39; and count(preceding-sibling::i[@type=&#39;s&#39;]) &gt; 0]&quot;&gt;
  12. &lt;!-- 什么都不做 --&gt;
  13. &lt;/xsl:when&gt;
  14. &lt;xsl:when test=&quot;current()[./@type=&#39;s&#39; and following-sibling::i[1][@type=&#39;r&#39;]]&quot;&gt;
  15. &lt;wrap&gt;
  16. &lt;xsl:apply-templates select=&quot;current() | //i[@type=&#39;r&#39; and preceding-sibling::i[@type=&#39;s&#39;]&quot; mode=&quot;fill&quot;/&gt;
  17. &lt;/wrap&gt;
  18. &lt;/xsl:when&gt;
  19. &lt;/xsl:choose&gt;
  20. &lt;/xsl:for-each&gt;

需要注意的是,后续的模板会处理所有的 &lt;I&gt; 节点。这些模板由于与问题本身无关而被省略了。

英文:

I have the following source XML

  1. &lt;root&gt;
  2. &lt;i type=&quot;r&quot;/&gt;
  3. &lt;i type=&quot;s&quot;/&gt;
  4. &lt;i type=&quot;r&quot;/&gt;
  5. &lt;i type=&quot;r&quot;/&gt;
  6. &lt;i type=&quot;s&quot;/&gt;
  7. &lt;i type=&quot;r&quot;/&gt;
  8. &lt;i type=&quot;s&quot;/&gt;
  9. &lt;/root&gt;

My goal is to wrap together all <i type="s"> occurrences with all immediate following-siblings (1..n) of type="r".

No wrapping for:

  • type="r" nodes w/o preceding type="s" node
  • type="s" nodes w/o immediate following type="r" node

Expected output:

  1. &lt;i type=&quot;r&quot;/&gt;
  2. &lt;wrap&gt;
  3. &lt;i type=&quot;s&quot;/&gt;
  4. &lt;i type=&quot;r&quot;/&gt;
  5. &lt;i type=&quot;r&quot;/&gt;
  6. &lt;/wrap&gt;
  7. &lt;wrap&gt;
  8. &lt;i type=&quot;s&quot;/&gt;
  9. &lt;i type=&quot;r&quot;/&gt;
  10. &lt;/wrap&gt;
  11. &lt;i type=&quot;s&quot;/&gt;

I have been trying to resolve the problem with the following XSLT snippet:

  1. &lt;xsl:for-each select=&quot;./i&quot;&gt;
  2. &lt;xsl:choose&gt;
  3. &lt;xsl:when test=&quot;current()[./@type=&#39;r&#39; and count(preceding-sibling::i[@type=&#39;s&#39;]) = 0]&quot;&gt;
  4. &lt;!-- Processing w/o wrap --&gt;
  5. &lt;xsl:apply-templates select=&quot;current()&quot; mode=&quot;fill&quot;/&gt;
  6. &lt;/xsl:when&gt;
  7. &lt;xsl:when test=&quot;current()[./@type=&#39;s&#39; and following-sibling::i[@type=&#39;s&#39;]]&quot;&gt;
  8. &lt;!-- Processing w/o wrap --&gt;
  9. &lt;xsl:apply-templates select=&quot;current()&quot; mode=&quot;fill&quot;/&gt;
  10. &lt;/xsl:when&gt;
  11. &lt;xsl:when test=&quot;current()[./@type=&#39;r&#39; and count(preceding-sibling::i[@type=&#39;s&#39;]) &gt; 0]&quot;&gt;
  12. &lt;!-- Do nothing --&gt;
  13. &lt;/xsl:when&gt;
  14. &lt;xsl:when test=&quot;current()[./@type=&#39;s&#39; and following-sibling::i[1][@type=&#39;r&#39;]]&quot;&gt;
  15. &lt;wrap&gt;
  16. &lt;xsl:apply-templates select=&quot;current() | //i[@type=&#39;r&#39; and preceding-sibling::i[@type=&#39;s&#39;]&quot; mode=&quot;fill&quot;/&gt;
  17. &lt;/wrap&gt;
  18. &lt;/xsl:when&gt;
  19. &lt;/xsl:choose&gt;
  20. &lt;/xsl:for-each&gt;

I'm constantly failing in defining the right context inside <wrap> </wrap>.

Need to note that subsequent templates are processing all <I> nodes. These templates have been omitted due to being irrelevant to the problem itself.

答案1

得分: 0

考虑以下示例:

XML

  1. <root>
  2. <i type="r">1</i>
  3. <i type="s">2</i>
  4. <i type="r">3</i>
  5. <i type="r">4</i>
  6. <i type="s">5</i>
  7. <i type="r">6</i>
  8. <i type="s">7</i>
  9. </root>

XSLT 1.0

  1. <xsl:stylesheet version="1.0"
  2. xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  3. <xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
  4. <xsl:strip-space elements="*"/>
  5. <xsl:key name="r" match="i[@type='r']" use="generate-id(preceding-sibling::i[@type='s'][1])" />
  6. <xsl:template match="/root">
  7. <xsl:copy>
  8. <xsl:copy-of select="key('r', '')"/>
  9. <xsl:for-each select="i[@type='s']">
  10. <xsl:variable name="r" select="key('r', generate-id())" />
  11. <xsl:choose>
  12. <xsl:when test="$r">
  13. <wrap>
  14. <xsl:copy-of select=". | $r"/>
  15. </wrap>
  16. </xsl:when>
  17. <xsl:otherwise>
  18. <xsl:copy-of select="."/>
  19. </xsl:otherwise>
  20. </xsl:choose>
  21. </xsl:for-each>
  22. </xsl:copy>
  23. </xsl:template>
  24. </xsl:stylesheet>

Result

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <root>
  3. <i type="r">1</i>
  4. <wrap>
  5. <i type="s">2</i>
  6. <i type="r">3</i>
  7. <i type="r">4</i>
  8. </wrap>
  9. <wrap>
  10. <i type="s">5</i>
  11. <i type="r">6</i>
  12. </wrap>
  13. <i type="s">7</i>
  14. </root>
英文:

Consider the following example:

XML

  1. &lt;root&gt;
  2. &lt;i type=&quot;r&quot;&gt;1&lt;/i&gt;
  3. &lt;i type=&quot;s&quot;&gt;2&lt;/i&gt;
  4. &lt;i type=&quot;r&quot;&gt;3&lt;/i&gt;
  5. &lt;i type=&quot;r&quot;&gt;4&lt;/i&gt;
  6. &lt;i type=&quot;s&quot;&gt;5&lt;/i&gt;
  7. &lt;i type=&quot;r&quot;&gt;6&lt;/i&gt;
  8. &lt;i type=&quot;s&quot;&gt;7&lt;/i&gt;
  9. &lt;/root&gt;

XSLT 1.0

  1. &lt;xsl:stylesheet version=&quot;1.0&quot;
  2. xmlns:xsl=&quot;http://www.w3.org/1999/XSL/Transform&quot;&gt;
  3. &lt;xsl:output method=&quot;xml&quot; version=&quot;1.0&quot; encoding=&quot;UTF-8&quot; indent=&quot;yes&quot;/&gt;
  4. &lt;xsl:strip-space elements=&quot;*&quot;/&gt;
  5. &lt;xsl:key name=&quot;r&quot; match=&quot;i[@type=&#39;r&#39;]&quot; use=&quot;generate-id(preceding-sibling::i[@type=&#39;s&#39;][1])&quot; /&gt;
  6. &lt;xsl:template match=&quot;/root&quot;&gt;
  7. &lt;xsl:copy&gt;
  8. &lt;xsl:copy-of select=&quot;key(&#39;r&#39;, &#39;&#39;)&quot;/&gt;
  9. &lt;xsl:for-each select=&quot;i[@type=&#39;s&#39;]&quot;&gt;
  10. &lt;xsl:variable name=&quot;r&quot; select=&quot;key(&#39;r&#39;, generate-id())&quot; /&gt;
  11. &lt;xsl:choose&gt;
  12. &lt;xsl:when test=&quot;$r&quot;&gt;
  13. &lt;wrap&gt;
  14. &lt;xsl:copy-of select=&quot;. | $r&quot;/&gt;
  15. &lt;/wrap&gt;
  16. &lt;/xsl:when&gt;
  17. &lt;xsl:otherwise&gt;
  18. &lt;xsl:copy-of select=&quot;.&quot;/&gt;
  19. &lt;/xsl:otherwise&gt;
  20. &lt;/xsl:choose&gt;
  21. &lt;/xsl:for-each&gt;
  22. &lt;/xsl:copy&gt;
  23. &lt;/xsl:template&gt;
  24. &lt;/xsl:stylesheet&gt;

Result

  1. &lt;?xml version=&quot;1.0&quot; encoding=&quot;UTF-8&quot;?&gt;
  2. &lt;root&gt;
  3. &lt;i type=&quot;r&quot;&gt;1&lt;/i&gt;
  4. &lt;wrap&gt;
  5. &lt;i type=&quot;s&quot;&gt;2&lt;/i&gt;
  6. &lt;i type=&quot;r&quot;&gt;3&lt;/i&gt;
  7. &lt;i type=&quot;r&quot;&gt;4&lt;/i&gt;
  8. &lt;/wrap&gt;
  9. &lt;wrap&gt;
  10. &lt;i type=&quot;s&quot;&gt;5&lt;/i&gt;
  11. &lt;i type=&quot;r&quot;&gt;6&lt;/i&gt;
  12. &lt;/wrap&gt;
  13. &lt;i type=&quot;s&quot;&gt;7&lt;/i&gt;
  14. &lt;/root&gt;

答案2

得分: 0

这是一个双键方法。

<xsl:key name="r" match="i[@type = 'r']" use="generate-id(preceding-sibling::i[@type = 's'][1])"/>
<xsl:key name="s" match="i[@type = 's']" use="generate-id(following-sibling::i[@type = 'r'][1])"/>

<xsl:template match="i[@type = 's']">
<xsl:variable name="id" select="generate-id(.)"/>
xsl:choose

<xsl:when test="key('r', $id)">
<xsl:element name="wrap">
<xsl:copy-of select="."/>

<xsl:for-each select="key('r', $id)">
<xsl:copy-of select="."/>
</xsl:for-each>
</xsl:element>
</xsl:when>
xsl:otherwise
<xsl:copy-of select="."/>
</xsl:otherwise>
</xsl:choose>
</xsl:template>

<xsl:template match="i[@type = 'r']">
<xsl:variable name="id" select="generate-id(.)"/>

<xsl:if test="not(key('s', $id))">
<xsl:copy-of select="."/>
</xsl:if>
</xsl:template>

<xsl:template match="node()">
<xsl:apply-templates select="node()"/>
</xsl:template>

英文:

Here is a two key approach.

  1. &lt;xsl:key name=&quot;r&quot; match=&quot;i[@type = &#39;r&#39;]&quot; use=&quot;generate-id(preceding-sibling::i[@type = &#39;s&#39;][1])&quot;/&gt;
  2. &lt;xsl:key name=&quot;s&quot; match=&quot;i[@type = &#39;s&#39;]&quot; use=&quot;generate-id(following-sibling::i[@type = &#39;r&#39;][1])&quot;/&gt;
  3. &lt;xsl:template match=&quot;i[@type = &#39;s&#39;]&quot;&gt;
  4. &lt;xsl:variable name=&quot;id&quot; select=&quot;generate-id(.)&quot;/&gt;
  5. &lt;xsl:choose&gt;
  6. &lt;!-- Does this s have following r&#39;s?--&gt;
  7. &lt;xsl:when test=&quot;key(&#39;r&#39;, $id)&quot;&gt;
  8. &lt;xsl:element name=&quot;wrap&quot;&gt;
  9. &lt;xsl:copy-of select=&quot;.&quot;/&gt;
  10. &lt;!-- Output the r&#39;s that immedately follow this s. --&gt;
  11. &lt;xsl:for-each select=&quot;key(&#39;r&#39;, $id)&quot;&gt;
  12. &lt;xsl:copy-of select=&quot;.&quot;/&gt;
  13. &lt;/xsl:for-each&gt;
  14. &lt;/xsl:element&gt;
  15. &lt;/xsl:when&gt;
  16. &lt;xsl:otherwise&gt;
  17. &lt;xsl:copy-of select=&quot;.&quot;/&gt;
  18. &lt;/xsl:otherwise&gt;
  19. &lt;/xsl:choose&gt;
  20. &lt;/xsl:template&gt;
  21. &lt;xsl:template match=&quot;i[@type = &#39;r&#39;]&quot;&gt;
  22. &lt;xsl:variable name=&quot;id&quot; select=&quot;generate-id(.)&quot;/&gt;
  23. &lt;!-- Only output r if it does not have a preceding s. --&gt;
  24. &lt;xsl:if test=&quot;not(key(&#39;s&#39;, $id))&quot;&gt;
  25. &lt;xsl:copy-of select=&quot;.&quot;/&gt;
  26. &lt;/xsl:if&gt;
  27. &lt;/xsl:template&gt;
  28. &lt;!-- Create a template to walk through the nodes on the document. --&gt;
  29. &lt;xsl:template match=&quot;node()&quot;&gt;
  30. &lt;xsl:apply-templates select=&quot;node()&quot;/&gt;
  31. &lt;/xsl:template&gt;

huangapple
  • 本文由 发表于 2023年2月16日 19:44:54
  • 转载请务必保留本文链接:https://go.coder-hub.com/75471782.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定