這兩天在處理支付金額校驗的時候出現了點問題,有個金額比較我用了BigDecimal的equals方法來比較兩個金額是否相等,結果導致金額比較出現錯誤(比如3.0與3.00的比較等)。
【注:以下所講都是以sun jdk 1.4.2版本為例,其他版本實現未必一致,請忽略】
首先看一下BigDecimal的equals方法:
1
2
3
4
5
6
7
|
public boolean equals(Object x){ if (!(x instanceof BigDecimal)) return false ; BigDecimal xDec = (BigDecimal) x; return scale == xDec.scale && intVal.equals(xDec.intVal); } |
可以看到BigDecimal的euquals方法是先判斷要比較的數據類型,如果對象類型一致前提下同時判斷精確度(scale)和值(BigInteger的equals方法)是否一致。
其實javadoc里面就已經寫的很明白:“Compares this BigDecimal with the specified Object for equality. Unlike compareTo, this method considers two BigDecimal objects equal only if they are equal in value and scale (thus 2.0 is not equal to 2.00 when compared by this method).”只是自己沒有去注意罷了!
再看一下compareTo方法:
1
2
3
4
5
6
7
8
9
10
11
12
|
public int compareTo(BigDecimal val){ /* Optimization: would run fine without the next three lines */ int sigDiff = signum() - val.signum(); if (sigDiff != 0) return (sigDiff > 0 ? 1 : -1); /* If signs match, scale and compare intVals */ BigDecimal arg[] = new BigDecimal[ 2 ]; arg[ 0 ] = this ; arg[ 1 ] = val; matchScale(arg); return arg[ 0 ].intVal.compareTo(arg[ 1 ].intVal); } |
可以看到這個方法里面有個matchScale的處理,意思是把精確度低的那個對象轉換為高精確度,然后再進行比較(同樣是BigInteger的compareTo方法),matchScale的實現如下:
1
2
3
4
5
6
|
private static void matchScale(BigDecimal[] val) { if (val[ 0 ].scale < val[ 1 ].scale) val[ 0 ] = val[ 0 ].setScale(val[ 1 ].scale); else if (val[ 1 ].scale < val[ 0 ].scale) val[ 1 ] = val[ 1 ].setScale(val[ 0 ].scale); } |
做個簡單測試:
1
2
3
|
System.out.println( new BigDecimal( "1.2" ).equals( new BigDecimal( "1.20" ))); //輸出false System.out.println( new BigDecimal( "1.2" ).compareTo( new BigDecimal( "1.20" )) == 0 ); //輸出true |
另外注意到我上面BigDecimal的構造方法里面傳入的都是字符串,如果傳入的是數字類型的話會有什么結果,大家可以自己測試一下,然后分析一下原因:
1
2
3
4
5
6
7
8
|
System.out.println( new BigDecimal( "1.2" ).equals( new BigDecimal( "1.20" ))); //輸出false System.out.println( new BigDecimal( "1.2" ).compareTo( new BigDecimal( "1.20" )) == 0 ); //輸出true System.out.println( new BigDecimal( 1.2 ).equals( new BigDecimal( "1.20" ))); //輸出是? System.out.println( new BigDecimal( 1.2 ).compareTo( new BigDecimal( "1.20" )) == 0 ); //輸出是? System.out.println( new BigDecimal( 1.2 ).equals( new BigDecimal( 1.20 ))); //輸出是? System.out.println( new BigDecimal( 1.2 ).compareTo( new BigDecimal( 1.20 )) == 0 ); //輸出是? |
最后結論是:對于BigDecimal的大小比較,用equals方法的話會不僅會比較值的大小,還會比較兩個對象的精確度,而compareTo方法則不會比較精確度,只比較數值的大小。
最后鄙視一下自己,用了這么多年的Java語言,連基本的常識都沒搞清楚!
以上這篇淺談java中BigDecimal的equals與compareTo的區別就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持服務器之家。