2017-02-23 31 views
0

我注意到了一些奇怪的東西。這是我的XMLXPath - Java XPath結果出人意料

<Items> 
    <Item> 
     <Name>A</Name> 
     <Amount>0.0012</Amount> 
     <Quantity>17</Quantity> 
     <TotalAmount>0.0204</TotalAmount> 
    </Item> 
    <Item> 
     <Name>B</Name> 
     <Amount>1</Amount> 
     <Quantity>2</Quantity> 
     <TotalAmount>2</TotalAmount> 
    </Item> 
    <Item> 
     <Name>C</Name> 
     <Amount>3</Amount> 
     <Quantity>2</Quantity> 
     <TotalAmount>6</TotalAmount> 
    </Item> 
</Items> 

這是我用

/項目/項目[((金額*數量)!=總金額)] /名稱

這中的XPath XPath必須打印其TotalAmount!=產品(數量,數量)的項目的名稱。

我得到的值A,但我不明白爲什麼會發生由於0.0012×17 = 0.0204

如果我刪除項目「A」,然後我沒有得到的結果。

也是如此的XPath的新版本以及

爲$ X IN /項目/項目[((金額*數量)!=總金額)返回 $ X /名稱

我在Java中使用Saxon 9。

有人可以解釋爲什麼會發生這種情況。

回答

1

嘗試使用xs:decimal/xs:integer更好的精度:/Items/Item[((xs:decimal(Amount) * xs:integer(Quantity)) != TotalAmount)]/Name

如果你看一下http://xsltransform.net/94AbWBV這確實

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

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

    <xsl:template match="/"> 
     <results> 
      <names-precise> 
      <xsl:value-of select="/Items/Item[((xs:decimal(Amount) * xs:integer(Quantity)) != TotalAmount)]/Name"/> 
      </names-precise> 
      <names-imprecise> 
       <xsl:value-of select="/Items/Item[((Amount * Quantity) != TotalAmount)]/Name"/> 
      </names-imprecise> 
      <xsl:apply-templates/> 
     </results> 

    </xsl:template> 

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

    <xsl:template match="Item"> 
     <xsl:copy> 
      <xsl:apply-templates/> 
      <double-computation> 
       <xsl:value-of select="Amount * Quantity"/> 
      </double-computation> 
      <decimal-computation> 
       <xsl:value-of select="xs:decimal(Amount) * xs:integer(Quantity)"/> 
      </decimal-computation> 
     </xsl:copy> 
    </xsl:template> 
</xsl:transform> 

你可以看到,使用不足以默認的浮點運算來計算確切的結果:

<results> 
    <names-precise/> 
    <names-imprecise>A</names-imprecise> 
    <Items> 
     <Item> 
     <Name>A</Name> 
     <Amount>0.0012</Amount> 
     <Quantity>17</Quantity> 
     <TotalAmount>0.0204</TotalAmount> 
     <double-computation>0.020399999999999998</double-computation> 
     <decimal-computation>0.0204</decimal-computation> 
     </Item> 
     <Item> 
     <Name>B</Name> 
     <Amount>1</Amount> 
     <Quantity>2</Quantity> 
     <TotalAmount>2</TotalAmount> 
     <double-computation>2</double-computation> 
     <decimal-computation>2</decimal-computation> 
     </Item> 
     <Item> 
     <Name>C</Name> 
     <Amount>3</Amount> 
     <Quantity>2</Quantity> 
     <TotalAmount>6</TotalAmount> 
     <double-computation>6</double-computation> 
     <decimal-computation>6</decimal-computation> 
     </Item> 
    </Items> 
</results> 

同樣不精確等存在使用IEEE雙數字格式的Javascript等語言:

document.getElementById('result').textContent = 0.0012 * 17;
<p>Result of <code>0.0012 * 17</code> with Javascript is <code id="result"></code>.</p>

+0

嗨馬丁。謝謝。像魅力一樣工作。 :) – Balu