BigDecimal详解 BigDecimal加减乘除运算 BigDecimal比较大小 BigDecimal保留两位小数

BigDecimal详解 BigDecimal加减乘除运算 BigDecimal比较大小 BigDecimal保留两位小数文章目录1、为什么要用BigDecimal?2、BigDecimal初始化赋值3、BigDecimal的加减乘除运算4、BigDecimal比较大小5、BigDecimal保留两位小数及舍入模式6、BigDecimal其他方法及常量1、为什么要用BigDecimal?工作中我们通过浮点数进行运算时,好像时不时的会出现一些小误差。例如:publicstaticvoidmain(String[]args){System.out.println(1.9-1.2);Sys

大家好,又见面了,我是你们的朋友全栈君。

1、为什么要用BigDecimal ?

工作中我们通过浮点数进行运算时,好像时不时的会出现一些小误差。例如:

public static void main(String[] args) { 
   
    System.out.println(1.9 - 1.2);
    System.out.println(1.9 - 1.5);
    System.out.println(100 - 99.8);
}

在这里插入图片描述
大致搜了一下原因,网上说我们的计算机是二进制的,而浮点数是没有办法通过二进制精准的表示出来。
也就导致在运算的时候,float类型和double类型很容易丢失精度。
所以在开发中,如果我们需要精确计算的结果,可以使用java.math包中提供的BigDecimal类来进行操作。

2、BigDecimal初始化赋值

方法 类型 描述
public BigDecimal(int val) 构造函数 int类型的值生成BigDecimal对象
public BigDecimal(long val) 构造函数 long类型的值生成BigDecimal对象
public BigDecimal(String val) 静态方法 String类型的值转换为BigDecimal类型
public static BigDecimal valueOf(double val) 静态方法 double类型的值转换为BigDecimal类型
public static BigDecimal valueOf(long val) 静态方法 long类型(包含int类型)的值转换为BigDecimal类型
  • 代码示例:
BigDecimal b = new BigDecimal("33");
BigDecimal c = BigDecimal.valueOf(4.7);
  • 注意:不建议使用public BigDecimal(double val)方式初始化值,编码时idea提示禁止使用构造方法BigDecimal(double),描述如下:
使用了new BigDecimal(double)构造函数 less... (Ctrl+F1) 
Inspection info: 
禁止使用构造方法BigDecimal(double)的方式把double值转化为BigDecimal对象 说明:反编译出的字节码文件显示每次循环都会new出一个StringBuilder对象,然后进行append操作,最后通过toString方法返回String对象,造成内存资源浪费。
            
Negative example(不建议使用):
    BigDecimal good1 = new BigDecimal(0.1);

Positive example(建议使用):
    BigDecimal good1 = new BigDecimal("0.1");
    BigDecimal good2 = BigDecimal.valueOf(0.1);

在这里插入图片描述

3、BigDecimal的加减乘除运算

运算法则 对应方法
加法 public BigDecimal add(BigDecimal value)
减法 public BigDecimal subtract(BigDecimal value)
乘法 public BigDecimal multiply(BigDecimal value)
除法 public BigDecimal divide(BigDecimal value)
  • 代码示例:
public static void main(String[] args) { 
   
    System.out.println("计算加法: " + BigDecimal.valueOf(1.9).add(BigDecimal.valueOf(0.2)));
    System.out.println("计算减法: " + BigDecimal.valueOf(1.9).subtract(BigDecimal.valueOf(1.5)));
    System.out.println("计算乘法: " + BigDecimal.valueOf(1.9).multiply(BigDecimal.valueOf(0.2)));
    System.out.println("计算除法: " + BigDecimal.valueOf(1.9).divide(BigDecimal.valueOf(0.2)));
}

在这里插入图片描述

  • 注意1:BigDecimal的运算结果都是返回了一个新的BigDecimal对象,并不是在原有的对象上进行操作。
public static void main(String[] args) { 
   
    BigDecimal a = BigDecimal.valueOf(5);
    System.out.println("a的地址:" + System.identityHashCode(a));
    a = a.add(BigDecimal.valueOf(3));
    System.out.println("计算后a的地址:" + System.identityHashCode(a));
}

在这里插入图片描述

  • 注意2:使用divide除法函数除不尽,出现无线循环小数的时候,就需要使用另外精确的小数位数以及舍入模式,不然会出现报错。例如:
public static void main(String[] args) { 
   
    BigDecimal a = BigDecimal.valueOf(10), b = BigDecimal.valueOf(3);
    System.out.println(a.divide(b));
}

// 该程序运行会出现以下错误
Exception in thread "main" java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result.
    at java.math.BigDecimal.divide(BigDecimal.java:1690)
    at com.fivesix._05_bigdecimal.Demo01.main(Demo01.java:31)

解决方法如下(此处舍入模式使用四舍五入的方式,其他模式在该文章后面有讲解):

public static void main(String[] args) { 
   
    BigDecimal a = BigDecimal.valueOf(10), b = BigDecimal.valueOf(3);
    System.out.println(a.divide(b, 3, BigDecimal.ROUND_HALF_UP));
}
// 该程序运行后输出:
3.33

4、BigDecimal比较大小

public int compareTo(BigDecimal val)

BigDecimal类提供的比较值的方法,注意比较的两个值均不能为空
a.compareTo(b)得到结果 1, 0, -1。

比较结果 描述
1 a 大于b
0 a 等于b
-1 a 小于b
  • 代码示例:
public static void main(String[] args) { 
   
    BigDecimal a = BigDecimal.valueOf(1);
    BigDecimal b = BigDecimal.valueOf(2);
    BigDecimal c = BigDecimal.valueOf(1);
    BigDecimal d = BigDecimal.ZERO;
    System.out.println("1和2比较结果:" + a.compareTo(b));
    System.out.println("1和1比较结果:" + a.compareTo(c));
    System.out.println("1和0比较判断:" + (a.compareTo(d) > 0) );
}

在这里插入图片描述

5、BigDecimal保留两位小数及舍入模式

public BigDecimal setScale(int newScale, int roundingMode)

用于格式化小数的方法,第一个值表示保留几位小数,第二个值表示格式化的类型。
8种类型如下:

格式化类型 描述
ROUND_DOWN 舍弃多余位数,如1.55会格式化为1.5,-1.55会格式化为-1.5
ROUND_UP 进位处理,如1.52会格式化为1.6,-1.52会格式化为-1.6
ROUND_HALF_UP 四舍五入,如果舍弃部分>= .5,则进位
ROUND_HALF_DOWN 五舍六入,如果舍弃部分> .5,则进位
ROUND_CEILING 正无穷大方向舍入模式。如果值为正数,则与ROUND_UP模式相同;如果值为负数,则与ROUND_DOWN模式相同
ROUND_FLOOR 负无穷大方向舍入模式。如果值为正数,则与ROUND_DOWN模式相同;如果值为负数,则与ROUND_UP模式相同
ROUND_UNNECESSARY 确认值的小数位数是否与传入第一个参数(保留小数的位数)相等,如果符合则返回值,如果不符抛出异常
ROUND_HALF_EVEN 如果舍弃部门左边的数字为奇数,则与ROUND_HALF_UP模式相同,如果为偶数则与ROUND_HALF_DOWN模式相同
  • 代码示例:
public static void main(String[] args) { 
   
    BigDecimal a = BigDecimal.valueOf(5.445);
    System.out.println("5.445舍弃多余位数:" + a.setScale(2, BigDecimal.ROUND_DOWN));
    System.out.println("5.445进位处理:" + a.setScale(2, BigDecimal.ROUND_UP));
    System.out.println("5.445四舍五入(舍弃部分>= .5,进位):" + a.setScale(2, BigDecimal.ROUND_HALF_UP));
    System.out.println("5.445四舍五入(舍弃部分未> .5,舍弃):" + a.setScale(2, BigDecimal.ROUND_HALF_DOWN));
    System.out.println("5.446四舍五入(舍弃部分> .5,进位):" + BigDecimal.valueOf(5.446).setScale(2, BigDecimal.ROUND_HALF_DOWN));
}

在这里插入图片描述

6、BigDecimal其他方法及常量

代码 类型 描述
BigDecimal.ZERO 常量 初始化一个为0的BigDecimal对象
BigDecimal.ONE 常量 初始化一个为1的BigDecimal对象
BigDecimal.TEN 常量 初始化一个为10的BigDecimal对象
public BigDecimal abs() 方法 求绝对值,不管正数还是负数,都得到正数
public BigDecimal negate() 方法 求相反数,正变负,负变正
public BigDecimal pow(int n) 方法 求乘方,如BigDecimal.valueOf(2).pow(3)的值为8
public BigDecimal max(BigDecimal val) 方法 两值比较,返回最大值
public BigDecimal min(BigDecimal val) 方法 两值比较,返回最小值
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/137201.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • LSTM和双向LSTM

    LSTM和双向LSTM文章目录1.RNN2.LSTM2.1简介2.2细胞状态2.3遗忘门2.4输入门2.5输出门3.双向LSTM(Bi-directionalLSTM)4.keras实现4.1LSTM模型4.2Bi-LSTM模型4.3训练模型4.4预测参考连接:https://www.cnblogs.com/Allen-rg/p/11712703.html1.RNN循环神经网络(RecurrentNeuralNetwork,RNN)是一种处理序列数据的网络。其中xt为t时刻的输入,h

    2022年6月16日
    39
  • 阿里用什么替代了dubbo_阿里面试必问题:Spring+MyBaits+微服务+Dubbo+Kakfa带解析

    前言很多同学在群里和我抱怨,面试的时候准备的不充分,导致面试结果不理想,也有很多同学苦于没有一份合适的面试指导。针对这些的同学,在这分享总结的Java面试的高频面试题(包括了Java集合,JVM,并发与多线程,Spring,MyBaits,微服务,Dubbo,Kakfa,中间件,Redis,数据库,设计模式等),进行了整理,免费分享给大家,希望大家能带着这些问题和答案解析,能让你进行有针对性行的学…

    2022年4月5日
    338
  • ffmpeg avformat_open_input返回失败,错误码-10049

    ffmpeg avformat_open_input返回失败,错误码-10049ffmpegavformat_open_input失败,返回错误码-10049,错误描述:Errornumber-10049occurred调试代码,找到如果设置了time_out参数,就会失败,不设置就没问题,具体原因还没有找到,备注此问题,以待后续继续分析源代码:av_register_all();avformat_network_init();AVFormatCo…

    2022年9月28日
    0
  • 使用Gstreamer处理RTSP视频流

    使用Gstreamer处理RTSP视频流文章目录RTSP视频流处理方法1.Gstreamer整体框架1.1MediaApplications1.2CoreFramework1.3Plugins2.Gstreamer组件2.1Element2.2Pad2.3Bin和Pipeline3.gstreamertools3.1gst-inspect-1.03.2gst-launch-1.04.参考链接RTSP视频流…

    2022年10月18日
    0
  • 文本挖掘的介绍

    文本挖掘的介绍1、文本挖掘的定义文本挖掘是指从大量文本的集合C中发现隐含的模式p。如果将C看作输入,将p看作输出,那么文本挖掘的过程就是从输入到输出的一个映射ξ:C→p。2、文本挖掘过程包含的技术文本特征的提取、信息检索、自然语言处理、文本挖掘、文本分类、文本聚类、关联分析等等3、文本挖掘的一般过程3.1 数据预处理技术预处理技术主要包括Stemming(英文)/分词(中文

    2022年6月29日
    17
  • Fuel 5.1安装openstack I版本号环境

    Fuel 5.1安装openstack I版本号环境

    2022年1月3日
    39

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号