国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發 > Java > 正文

因BigDecimal類型數據引出的問題詳析

2024-07-14 08:42:04
字體:
來源:轉載
供稿:網友

前言

我們都知道,java中對大小數,高精度的計算都會用到BigDecimal.但是在實際應用中,運用BigDecimal還是會遇到一些問題,下面話不多說了,來一起看看詳細的介紹吧

問題描述:

程序中需要判斷一個字段是否為0(字段類型為BigDecimal),想都沒想,對象的判斷用equals?結果卻與預期有一定的差距,看下面代碼及運行結果。

 public static void main(String[] args) {  BigDecimal decimal1 = BigDecimal.valueOf(0);  BigDecimal decimal2 = new BigDecimal("0.00");  System.out.println("the result is " +decimal1.equals(decimal2)); }

運行結果:

the result is false

結論: BigDecimal類型比較相等不能簡單的通過equals方法實現。

BigDecimal類的equals方法源碼如下:

 public boolean equals(Object x) {  if (!(x instanceof BigDecimal))   return false;  BigDecimal xDec = (BigDecimal) x;  if (x == this)   return true;  if (scale != xDec.scale)//這里會比較數字的精度   return false;  long s = this.intCompact;  long xs = xDec.intCompact;  if (s != INFLATED) {   if (xs == INFLATED)    xs = compactValFor(xDec.intVal);   return xs == s;  } else if (xs != INFLATED)   return xs == compactValFor(this.intVal);  return this.inflate().equals(xDec.inflate()); }

看上面的注釋可以知道,BigDecimal類的equals方法會判斷數字的精度,看下面的代碼及運行結果:

 public static void main(String[] args) {  BigDecimal decimal1 = BigDecimal.valueOf(0).setScale(2);  BigDecimal decimal2 = new BigDecimal("0.00").setScale(2);  System.out.println("the result is " +decimal1.equals(decimal2)); }

運行結果:

the result is true

結論: 使用BigDecimal類equals方法判斷兩個BigDecimal類型的數據時,需要設置精度,否則結果可能不正確。

思考:每次都設置精度比較麻煩,有其他方式進行相等的比較嗎?

看了下BigDecimal的方法列表,有一個名為compareTo的方法,通過注釋可知,貌似可以進行不同精度的比較,看下面的代碼。

 public static void main(String[] args) {  BigDecimal decimal1 = BigDecimal.valueOf(1.1);  BigDecimal decimal2 = new BigDecimal("1.10");  System.out.println("the result is " +decimal1.compareTo(decimal2)); }

運行結果:

the result is 0

0表示兩個數相等,所有可以通過compareTo實現不同精度的兩個BigDecimal類型的數字是否相等的比較

引出的問題:公司的項目中,為了避免由于精度丟失引起問題,凡是有精度要求的字段用的都是BigDecimal類型。數據持久層用的是Mybatis框架,Mybatis的mapper文件中有些條件判斷用的是BigDecimal對應的字段,如下:

<select id="selectByCondition" resultType="com.scove.demo.domain.Score"> select * from tb_score where 1=1  <if test="score!=null and score!=0">  and score>#{score} </if> ...

score是一個BigDecimal類型的字段,score!=0 Mybatis是如何進行判斷的,會不會用的是上面的equals方法?如果是那么項目上線會不會捅大簍子,想到這兒,有點怕了。寫了個程序測了下,這樣寫完全沒問題,能夠達到我想要的目的。但是還是有點擔心,看看Mybatis底層是如何實現的吧,以免以后犯類似的錯誤嘛。

經過分析調試,很快就找到了關鍵代碼的位置,如下:

public class ExpressionEvaluator { public boolean evaluateBoolean(String expression, Object parameterObject) { Object value = OgnlCache.getValue(expression, parameterObject); if (value instanceof Boolean) {  return (Boolean) value; } if (value instanceof Number) {  return !new BigDecimal(String.valueOf(value)).equals(BigDecimal.ZERO); } return value != null; }...
public final class OgnlCache {.... public static Object getValue(String expression, Object root) { try {  Map<Object, OgnlClassResolver> context = Ognl.createDefaultContext(root, new OgnlClassResolver());  return Ognl.getValue(parseExpression(expression), context, root); } catch (OgnlException e) {  throw new BuilderException("Error evaluating expression '" + expression + "'. Cause: " + e, e); } }

用的是表達式求值,Ognl這個類的竟然沒有源碼,apache的官網上找了下,是有相應的源碼的,只是需要單獨下載,真麻煩,算了不看了。據說底層用的是Spring 的ognl表達式,我也不 關心了。但是以后如果不確定mapper中的test是否正確咋個辦?

想了下,還是寫一個工具類在拿不準的時候用一下吧,反正拿不準的時候肯定很少,所以隨便寫了簡單的吧,如下:

 public static void main(String[] args) {  ExpressionEvaluator evaluator = new ExpressionEvaluator();  String expression = "score!=null and score!=0";  DynamicContext context = new DynamicContext(new Configuration(), null);  context.bind("score", BigDecimal.valueOf(0.1));  Boolean flag = evaluator.evaluateBoolean(expression , context.getBindings());  System.out.println("the result is " +flag); }

運行結果:

the result is true

總結

開發過程中,一定要細心去處理細節上的東西,不然一不小心就跳坑里了,輕則系統造成數據的不一致,修復數據;重則造成重大的損失....

好了,以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VeVb武林網的支持。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 攀枝花市| 新巴尔虎右旗| 朝阳县| 静海县| 托克逊县| 贵定县| 昆明市| 墨江| 白银市| 广昌县| 库车县| 宁化县| 红原县| 勐海县| 合作市| 河北省| 贡觉县| 太康县| 吉隆县| 商都县| 玉屏| 贵港市| 明溪县| 达拉特旗| 仁化县| 禹州市| 故城县| 佛冈县| 策勒县| 三穗县| 信阳市| 南岸区| 二手房| 桦南县| 郸城县| 合肥市| 迭部县| 正宁县| 竹溪县| 章丘市| 中西区|