通过 xslt 从源头重复考虑

Posted

技术标签:

【中文标题】通过 xslt 从源头重复考虑【英文标题】:duplicate consideration from source through xslt 【发布时间】:2021-06-21 23:09:25 【问题描述】:

输入数据:以下是从源获取的输入数据。我们如何通过 XSLT 映射 SCPI 来实现,在此先感谢您,并期待您的回复。 我有以下问题(在这个领域经常讨论,但我仍然无法得到确切的解决方案):我有一个如下的 XML:

         <orderHeader>
            <distributionChannelCode>10</distributionChannelCode>
            <orderItem>
                <productCode>AQ7481-002-001</productCode>
            </orderItem>
              <orderItem>
                <productCode>AQ7481-002-004</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-001</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-003</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-002</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-003</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-003</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-004</productCode>
            </orderItem>
             <orderItem>
                <productCode>AQ7481-002-002</productCode>
            </orderItem>
        </orderHeader>

所需输出:

<orderHeader>
            <distributionChannelCode>10</distributionChannelCode>
            <orderItem>
                <productCode>AQ7481-002-001</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-002</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-003</productCode>
            </orderItem>
            <orderItem>
                <productCode>AQ7481-002-004</productCode>
            </orderItem>
        </orderHeader>  
        <orderHeader>
            <distributionChannelCode>10</distributionChannelCode>
            <orderItem>
                <productCode>AQ7481-002-001</productCode>
            </orderItem>
             <orderItem>
                <productCode>AQ7481-002-002</productCode>
            </orderItem>
             <orderItem>
                <productCode>AQ7481-002-003</productCode>
            </orderItem>
        </orderHeader>
        <orderHeader>
            <distributionChannelCode>10</distributionChannelCode>
             <orderItem>
                <productCode>AQ7481-002-003</productCode>
            </orderItem>
        </orderHeader>

【问题讨论】:

【参考方案1】:

我不确定我是否理解了其中的逻辑,但可能以下内容按 productCode 对 orderItems 进行分组以确定哪个组的项目最多,然后输出与每个组中的相应项目一样多的 orderHeaders 来完成这项工作:

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    exclude-result-prefixes="#all"
    version="3.0">

  <xsl:mode on-no-match="shallow-copy"/>

  <xsl:output method="xml" indent="yes"/>
  <xsl:strip-space elements="*"/>

  <xsl:template match="orderHeader">
      <xsl:variable name="groups" as="map(xs:string, element(orderItem)*)*">
          <xsl:for-each-group select="orderItem" group-by="string(productCode)">
              <xsl:map-entry key="current-grouping-key()" select="current-group()"/>
          </xsl:for-each-group>
      </xsl:variable>
      <xsl:variable name="this" select="."/>
      <xsl:variable name="max-items" select="max($groups!count(?*))"/>
      <xsl:iterate select="1 to $max-items">
          <orderHeader>
              <xsl:copy-of 
                select="$this/distributionChannelCode, 
                        $groups ! ?*[current()]"/>
          </orderHeader>
      </xsl:iterate>
  </xsl:template>
  
</xsl:stylesheet>

【讨论】:

非常感谢 :),马丁。代码运行良好,但能否请您在 XSLT 2.0 中提供相同的代码——因为我熟悉 2.0 版并且我对 3.0 版非常陌生。非常感谢我 嗯,有文档和培训材料,如altova.com/training/xpath3/xpath-31#maps、w3.org/TR/xslt-30/#iterate、w3.org/TR/xpath-31/#id-maps。我没有使用 XSLT 3 的特性来使事情复杂化,如果您可以将组存储在一系列地图中并进一步处理,问题会更容易。 我从上周开始尝试这个逻辑,你终于做到了。非常感谢您的快速回复.. 你好马丁,如果值像 A123-100-200 一样,上面的代码不起作用。你能回复一下吗。我们如何实现这一目标 @srikanthboda,用最少但完整的输入样本、想要的结果、您尝试过的代码以及得到的错误结果提出一个新问题。

以上是关于通过 xslt 从源头重复考虑的主要内容,如果未能解决你的问题,请参考以下文章

需要 XSLT 转换以删除重复元素 - 按属性排序

使用 XSLT 进行空前缀转换的 XML [重复]

自动驾驶网络系列二: 从哲学源头开始思考架构设计

Xslt在通过xsl文件解析时为负值添加括号'()'作为响应

XSLT 3.0 - 在 XSLT 3.0 xml-to-json() 中出现错误“重复键值”

从 javascript 调用 XSLT