XSLT 1.0 Tansfomation - перемещение данных родного брата в определенные родственные братья - PullRequest
3 голосов
/ 06 января 2012

У меня проблемы с разработкой преобразования xslt, и я был бы очень признателен за помощь. Я потратил довольно много времени, используя различные методы в XPath и XQuery. Кроме того, я ограничен xslt 1.0.

Преобразование включает в себя внесение изменений в позиции продукта в файле заказа xml. Исходный файл XML содержит элементы, однако некоторые из элементов строки являются ссылками на купоны на скидку (см. Dsc-102 и dsc-133 ниже). Что мне нужно сделать, так это удалить узлы orderDetails для ссылок на купоны на скидку и добавить содержащую информацию к соответствующим элементам их родственных продуктов (см. Образец Transformed XML ниже). Каждая ссылка на купон на скидку указывает соответствующие позиции товара в конце названия товара (например,…. [Glv-001] [glv-003]).

Исходный файл XML - Ниже приведен оригинальный XML-файл, который содержит 1 заказ с 3 наименованиями товаров и 2 ссылками на купоны на скидку. Ссылка на скидку 'dsc-102' соответствует 2 товарным позициям 'glv-001' и 'glv-003'. Ссылка на скидку 'dsc-133' соответствует 1 позиции товара 'sho-123'.

<xmldata>
<Order>
    <orderID>1010</orderID>
    <custFirstName>Jim</custFirstName>
    <custLastName>Jones</custLastName>
    <orderDetails>
        <productCode>sho-123</productCode>
        <productName>Leather Windsor Shoes - size 10</productName>
    </orderDetails>
    <orderDetails>
        <productCode>glv-001</productCode>
        <productName>Leather gloves - size Small</productName>
    </orderDetails>
    <orderDetails>
        <productCode>glv-003</productCode>
        <productName>Leather gloves - size XLarge</productName>
    </orderDetails>
    <orderDetails>
        <productCode>dsc-102</productCode>
        <productName>10% Discount for Leather Gloves [glv-001][glv-003]</productName>
    </orderDetails>
    <orderDetails>
        <productCode>dsc-133</productCode>
        <productName>Free Shipping for Windsor Shoes [sho-123]</productName>
    </orderDetails>
</Order>

Преобразованный файл XML - Ниже приведен преобразованный XML, которого я хочу достичь. Передача удалила обе ссылки на купоны на скидку и добавила узел "discountCoupon" к соответствующим элементам их родственного товара.

<xmldata>
<Orders>
    <orderID>1010</orderID>
    <custFirstName>Jim</custFirstName>
    <custLastName>Jones</custLastName>
    <orderDetails>
        <productCode>sho-123</productCode>
        <productName>Leather Windsor Shoes - size 10</productName>
        <discountCoupon>Free Shipping for Windsor Shoes</discountCoupon>
    </orderDetails>
    <orderDetails>
        <productCode>glv-001</productCode>
        <productName>Leather gloves - size Small</productName>
        <discountCoupon>10% Discount for Leather Gloves</discountCoupon>
    </orderDetails>
    <orderDetails>
        <productCode>glv-003</productCode>
        <productName>Leather gloves - size XLarge</productName>
        <discountCoupon>10% Discount for Leather Gloves</discountCoupon>
    </orderDetails>
</Orders>

Что я уже пробовал - Честно говоря, у меня был довольно ограниченный успех с этой проблемой. Самое близкое, что у меня есть, было следующее. Тем не менее, это было довольно далеко от моего предполагаемого результата, и «совпадения» - это функция XLST 2.0, и я ограничен версией 1.

<xsl:if test="../OrderDetails[ProductCode = 'DSC-15'] and matches(ProductCode,'AH010585059',i)">DiscountCoupon</xsl:if>

Если бы кто-то мог мне помочь с этой проблемой или дать толчок в правильном направлении, я был бы очень признателен.

-Cheers.

Ответы [ 2 ]

2 голосов
/ 06 января 2012

Решение, похожее на @lwburk, но более простое - без <xsl:if> и без <xsl:variable>:

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

 <xsl:template match=
  "orderDetails[not(starts-with(productCode, 'dsc-'))]">
  <xsl:copy>
   <xsl:apply-templates select="node()|@*"/>
   <xsl:apply-templates mode="coupon" select=
   "../orderDetails[starts-with(productCode, 'dsc-')]
                    [contains(productName,
                           concat('[', current()/productCode, ']')
                          )
                     ]/productName
   "/>
  </xsl:copy>
 </xsl:template>

 <xsl:template match="orderDetails[starts-with(productCode, 'dsc-')]"/>

 <xsl:template match="productName" mode="coupon">
  <discountCoupon>
   <xsl:value-of select="substring-before(., ' [')"/>
  </discountCoupon>
 </xsl:template>
</xsl:stylesheet>

при применении этого преобразования кпредоставленный документ XML :

<Order>
    <orderID>1010</orderID>
    <custFirstName>Jim</custFirstName>
    <custLastName>Jones</custLastName>
    <orderDetails>
        <productCode>sho-123</productCode>
        <productName>Leather Windsor Shoes - size 10</productName>
    </orderDetails>
    <orderDetails>
        <productCode>glv-001</productCode>
        <productName>Leather gloves - size Small</productName>
    </orderDetails>
    <orderDetails>
        <productCode>glv-003</productCode>
        <productName>Leather gloves - size XLarge</productName>
    </orderDetails>
    <orderDetails>
        <productCode>dsc-102</productCode>
        <productName>10% Discount for Leather Gloves [glv-001][glv-003]</productName>
    </orderDetails>
    <orderDetails>
        <productCode>dsc-133</productCode>
        <productName>Free Shipping for Windsor Shoes [sho-123]</productName>
    </orderDetails>
</Order>

желаемый, правильный результат получен :

<Order>
   <orderID>1010</orderID>
   <custFirstName>Jim</custFirstName>
   <custLastName>Jones</custLastName>
   <orderDetails>
      <productCode>sho-123</productCode>
      <productName>Leather Windsor Shoes - size 10</productName>
      <discountCoupon>Free Shipping for Windsor Shoes</discountCoupon>
   </orderDetails>
   <orderDetails>
      <productCode>glv-001</productCode>
      <productName>Leather gloves - size Small</productName>
      <discountCoupon>10% Discount for Leather Gloves</discountCoupon>
   </orderDetails>
   <orderDetails>
      <productCode>glv-003</productCode>
      <productName>Leather gloves - size XLarge</productName>
      <discountCoupon>10% Discount for Leather Gloves</discountCoupon>
   </orderDetails>
</Order>

Пояснение : правильное использованиеи переопределение правила идентификации и шаблонов / сопоставление с образцом.

2 голосов
/ 06 января 2012

Следующая таблица стилей дает правильный результат:

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:output indent="yes" omit-xml-declaration="yes"/>
    <xsl:strip-space elements="*"/>
    <xsl:template match="@*|node()">
        <xsl:copy>
            <xsl:apply-templates select="@*|node()"/>
        </xsl:copy>
    </xsl:template>
    <xsl:template match="orderDetails[not(starts-with(productCode, 'dsc'))]">
        <xsl:copy>
            <xsl:apply-templates select="@*|node()"/>
            <xsl:variable name="discount"
                select="../orderDetails[starts-with(productCode, 'dsc') and 
                      contains(productName, 
                         concat('[', current()/productCode, ']'))]/productName"/>
            <xsl:if test="$discount">
                <discountCoupon>
                    <xsl:value-of select="substring-before($discount, ' [')"/>
                </discountCoupon>
            </xsl:if>
        </xsl:copy>
    </xsl:template>
    <xsl:template match="orderDetails[starts-with(productCode, 'dsc')]"/>
</xsl:stylesheet>

Примечания и пояснения:

  • Преобразование идентичности копирует большинство узлов без изменений
  • Скидки копируются в недисконтные элементы по мере их обработки
  • Скидка orderDetails игнорируется
...