我在处理xslt转换时遇到了问题,我非常感谢您的帮助。我花了很多时间在XPath和XQuery中使用各种方法。此外,我被限制为xslt 1.0。
转换涉及对xml订单文件中的产品项进行更改。原始XML文件中有Items,但是其中一些行Items是折扣优惠券引用(请参见下面的dsc-102和dsc-133)。我需要实现的是删除折扣优惠券引用的"orderDetails"节点,并将包含的信息添加到相应的同级产品项中(请参阅下面的Transformed XML示例)。每个折扣券参考在其产品名称的末尾指定其相应的产品项目(例如……[glv-001][glv-003])。
原始XML文件 -以下是原始XML文件,其中包含1个订单、3个产品项目和2个折扣优惠券参考。折扣参考"dsc-102"对应于两个产品项目"glv-001"one_answers"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>
到目前为止我尝试了什么-老实说,我在这个问题上取得的成功非常有限。我最接近它的是以下内容。然而,这与我的预期结果相去甚远,"matches"是XLST 2.0函数,我仅限于版本1。
<xsl:if test="../OrderDetails[ProductCode = 'DSC-15'] and matches(ProductCode,'AH010585059',i)">DiscountCoupon</xsl:if>
如果有人能帮我解决这个问题,或者朝着正确的方向推动我,我会非常感激
-干杯。
类似于@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>
解释:适当使用和覆盖标识规则以及模板/模式匹配。
以下样式表生成正确的结果:
<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
被忽略