0

以下のサンプルxmlでは、ASNOutDescに複数のASNOutDistroを含めることができ、各ASNOutDistroに複数のASNOutCtnを含めることができます。各ASNOutCtnの下に複数のASNOutItemを含めることができます。各ASNOutCtnの下のすべてのアイテムをマージし、unit_qtyを追加する必要があります。Iamは以下のxsltを試していますが、それはmergです

入力XML:

<ASNoutDesc>
  <ASNOutDesc>
    <to_location>14</to_location>
    <from_location>777</from_location>
    <asn_nbr>Bol1</asn_nbr>
    <container_qty>3</container_qty>
    <shipment_date>2012-04-20T15:37:30.757</shipment_date>
    <ASNOutDistro>
      <distro_nbr>101</distro_nbr>
      <distro_doc_type>A</distro_doc_type>
      <ASNOutCtn>
        <container_id>1</container_id>
        <ASNOutItem>
          <item_id>item-1</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
        <ASNOutItem>
          <item_id>item-1</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
      </ASNOutCtn>
    </ASNOutDistro>
    <ASNOutDistro>
      <distro_nbr>102</distro_nbr>
      <distro_doc_type>A</distro_doc_type>
      <ASNOutCtn>
        <container_id>2</container_id>
        <ASNOutItem>
          <item_id>item-1</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
        <ASNOutItem>
          <item_id>item-2</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
        <ASNOutItem>
          <item_id>item-2</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
      </ASNOutCtn>
      <ASNOutCtn>
        <container_id>3</container_id>
        <ASNOutItem>
          <item_id>item-1</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
        <ASNOutItem>
          <item_id>item-2</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
      </ASNOutCtn>
    </ASNOutDistro>
  </ASNOutDesc>
</ASNoutDesc>

必要な出力:

<ASNoutDesc>
  <ASNOutDesc>
    <to_location>14</to_location>
    <from_location>777</from_location>
    <asn_nbr>Bol1</asn_nbr>
    <container_qty>3</container_qty>
    <shipment_date>2012-04-20T15:37:30.757</shipment_date>
    <ASNOutDistro>
      <distro_nbr>101</distro_nbr>
      <distro_doc_type>A</distro_doc_type>
      <ASNOutCtn>
        <container_id>1</container_id>
        <ASNOutItem>
          <item_id>item-1</item_id>
          <unit_qty>4</unit_qty>
        </ASNOutItem>
      </ASNOutCtn>
    </ASNOutDistro>
    <ASNOutDistro>
      <distro_nbr>102</distro_nbr>
      <distro_doc_type>A</distro_doc_type>
      <ASNOutCtn>
        <container_id>2</container_id>
        <ASNOutItem>
          <item_id>item-1</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
        <ASNOutItem>
          <item_id>item-2</item_id>
          <unit_qty>4</unit_qty>
        </ASNOutItem>
      </ASNOutCtn>
      <ASNOutCtn>
        <container_id>3</container_id>
        <ASNOutItem>
          <item_id>item-1</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
        <ASNOutItem>
          <item_id>item-2</item_id>
          <unit_qty>2</unit_qty>
        </ASNOutItem>
      </ASNOutCtn>
    </ASNOutDistro>
  </ASNOutDesc>
</ASNoutDesc>

上に示したように、各ASNOutDistroの下の各ASNOutCtnのアイテムをマージする必要があります。

XSLT Iamの使用:

<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="kByI" match="ASNOutItem" use="item_id"/>
  <xsl:template match="node()|@*">
    <xsl:param name="pNewValueQty"/>
    <xsl:copy>
      <xsl:apply-templates select="node()|@*">
        <xsl:with-param name="pNewValueQty" select="$pNewValueQty"/>
      </xsl:apply-templates>
    </xsl:copy>
  </xsl:template>
  <xsl:template match="ASNOutItem[generate-id()=generate-id(key('kByI',item_id)[1])]">
    <xsl:copy>
      <xsl:apply-templates select="node()|@*">
        <xsl:with-param name="pNewValueQty" select="sum(key('kByI',item_id)/unit_qty)"/>
      </xsl:apply-templates>
    </xsl:copy>
  </xsl:template>
  <xsl:template match="ASNOutItem"/>
  <xsl:template match="unit_qty/text()">
    <xsl:param name="pNewValueQty"/>
    <xsl:value-of select="$pNewValueQty"/>
  </xsl:template>
</xsl:stylesheet>

私が得た出力:

<ASNoutDesc>
  <ASNOutDesc>
    <to_location>14</to_location>
    <from_location>777</from_location>
    <asn_nbr>Bol1</asn_nbr>
    <container_qty>3</container_qty>
    <shipment_date>2012-04-20T15:37:30.757</shipment_date>
    <ASNOutDistro>
      <distro_nbr>101</distro_nbr>
      <distro_doc_type>A</distro_doc_type>
      <ASNOutCtn>
        <container_id>1</container_id>
        <ASNOutItem>
          <item_id>item-1</item_id>
          <unit_qty>8</unit_qty>
        </ASNOutItem>
      </ASNOutCtn>
    </ASNOutDistro>
    <ASNOutDistro>
      <distro_nbr>102</distro_nbr>
      <distro_doc_type>A</distro_doc_type>
      <ASNOutCtn>
        <container_id>2</container_id>
        <ASNOutItem>
          <item_id>item-2</item_id>
          <unit_qty>6</unit_qty>
        </ASNOutItem>
      </ASNOutCtn>
      <ASNOutCtn>
        <container_id>3</container_id>
      </ASNOutCtn>
    </ASNOutDistro>
  </ASNOutDesc>
</ASNoutDesc>

-私のコードはxml内のすべてのアイテムをマージしています。必要な出力を取得するのを手伝ってください。

前もって感謝します。

4

2 に答える 2

0

あなたが間違っていたのは、ASNOutItem ノードのグループ化で親ノードを考慮するのを忘れたことです。グループ化は ASNOutCtn 境界を越えないため、キー定義は、item_id と ASNOutItem の親である ASNOutCtn ノードの結合をキーにする必要があります。

このスタイルシートはあなたが望むものです...

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:key name="kByI" match="ASNOutItem" use="concat(item_id,'+',generate-id(parent::*))"/>

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

<xsl:template match="ASNOutCtn">
 <xsl:copy>
  <xsl:apply-templates select="@*"/>
  <xsl:apply-templates select="node()[not(self::ASNOutItem)]"/>
  <xsl:apply-templates
    select="ASNOutItem[
     generate-id(.) =
     generate-id( key('kByI',concat(item_id,'+',generate-id(parent::*)))[1])]"
    mode="merge"/>
 </xsl:copy>
</xsl:template>

<xsl:template match="ASNOutItem" mode="merge">
 <xsl:copy>
  <xsl:apply-templates select="@*"/>
  <xsl:apply-templates select="node()[not(self::unit_qty)]"/>
  <unit_qty>
    <xsl:value-of select="sum(
      key('kByI',concat(item_id,'+',generate-id(parent::*)))/unit_qty)" />
  </unit_qty>
 </xsl:copy>
</xsl:template>

</xsl:stylesheet>
于 2012-07-13T17:56:53.620 に答える
-1

なぜ使用しないのsum(ASNOutItem/item_id)ですか?

申し訳ありませんが、私の意見では、そのコードを1年で見ると、自分が何をしたか、何をしているのかを理解するのにかなりの時間がかかります. 私は次のようなことをしたいと思います:

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<ASNoutDesc>
    <ASNOutDesc>
        <xsl:copy-of select="*[name() != 'ASNOutDistro']"/>
        <xsl:for-each select="ASNOutDistro">
            <xsl:copy-of select="*[name() != 'ASNOutCtn']"/>
            <xsl:for-each select="ASNOutCtn">
                <ASNOutCtn>
                    <xsl:copy-of select="*[name() != 'ASNOutItem']"/>
                    <ASNOutItem>
                        <xsl:copy-of select="ASNOutItem[1]/*[name() != 'unit_qty']"/>
                        <unit_qty>
                            <xsl:value-of select="sum(ASNOutItem/*/unit_qty)"/>
                        </unit_qty>
                    </ASNOutItem>
                </ASNOutCtn>
            </xsl:for-each>
        </xsl:for-each>
    </ASNOutDesc>
</ASNoutDesc>

于 2014-03-03T08:45:53.593 に答える