浅谈java中BigDecimal的equals与compareTo的区别

作者:jingxian 时间:2023-09-02 07:20:22 

这两天在处理支付金额校验的时候出现了点问题,有个金额比较我用了BigDecimal的equals方法来比较两个金额是否相等,结果导致金额比较出现错误(比如3.0与3.00的比较等)。

【注:以下所讲都是以sun jdk 1.4.2版本为例,其他版本实现未必一致,请忽略】

首先看一下BigDecimal的equals方法:


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方法:


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的实现如下:


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);
 }

做个简单测试:


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的构造方法里面传入的都是字符串,如果传入的是数字类型的话会有什么结果,大家可以自己测试一下,然后分析一下原因:


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语言,连基本的常识都没搞清楚!

标签:compareto,equals,java
0
投稿

猜你喜欢

  • Android ViewPager实现Banner循环播放

    2022-07-16 06:24:03
  • Spring实现动态切换多数据源的解决方案

    2023-05-21 13:07:15
  • SSH框架网上商城项目第24战之Struts2中处理多个Model请求的方法

    2023-02-14 20:49:21
  • Flutter源码分析之自定义控件(RenderBox)指南

    2022-09-08 21:17:28
  • Android实现可拖动层叠卡片布局

    2023-03-25 03:28:06
  • 深度剖析java动态静态代理原理源码

    2021-10-25 08:10:31
  • C语言代码实现三子棋小游戏

    2023-04-16 02:12:41
  • java计算两个时间相差天数的方法汇总

    2023-08-29 23:39:03
  • Mybatis-plus配置分页插件返回统一结果集

    2022-05-27 19:15:06
  • Android仿微信QQ设置图形头像裁剪功能

    2022-06-21 10:12:25
  • C# 读写自定义的Config文件的实现方法

    2022-09-08 23:22:35
  • C# 调用Delphi dll 实例代码

    2023-06-12 22:46:06
  • 安卓模拟器genymotion的安装与使用图文教程

    2021-12-13 19:28:36
  • Android利用Sensor(传感器)实现指南针小功能

    2021-12-27 14:22:36
  • Spring MVC数据绑定方式

    2023-03-18 03:06:55
  • Java实现XML文件学生通讯录

    2023-07-23 19:21:49
  • 关于C# dynamic装箱问题

    2021-09-13 19:22:40
  • 详解Android TextView属性ellipsize多行失效的解决思路

    2022-03-02 16:00:32
  • Android实现点击切换视图并跳转传值

    2022-07-14 07:23:28
  • Maven打包没有指定主类问题(xxx.jar中没有主清单属性)

    2023-11-26 21:34:44
  • asp之家 软件编程 m.aspxhome.com