Java之BigDecimal使用

news2024/11/24 7:50:43

Java之BigDecimal使用

1、BigDecimal概述

​ BigDecimal用来对超过16位有效位的数进行精确的运算。双精度浮点型变量double可以处理16位有效数,但在实际应用中,可能需要对更大或者更小的数进行运算和处理。一般情况下,对于那些不需要准确计算精度的数字,我们可以直接使用Float和Double处理,但是Double.valueOf(String) 和Float.valueOf(String)会丢失精度。所以在开发中,如果你所做的业务跟财务相关需要精确计算的结果,那必须使用BigDecimal类来操作啦!

2、常用构造器及方法描述

2.1、常用构造器

  • BigDecimal(int) 创建一个具有参数所指定整数值的对象
  • BigDecimal(double) 创建一个具有参数所指定双精度值的对象 【不推荐使用】
  • BigDecimal(long) 创建一个具有参数所指定长整数值的对象
  • BigDecimal(String) 创建一个具有参数所指定以字符串表示的数值的对象。【推荐使用】
2.1.1、不推荐使用double类型构造器的原因
public static void main(String[] args) {
    BigDecimal decimal = new BigDecimal(0.111);
    BigDecimal decimal1 = new BigDecimal("0.111");
    System.out.println(decimal);
    System.out.println(decimal1);
}

在这里插入图片描述

看到上述结果发现使用 double 为参数的构造器会出现不可预知的结果,idea也会提示我们不要使用double类型的构造器构造 BigDecimal对象

原因描述

  1. 参数类型为double的构造方法的结果有一定的不可预知性。有人可能认为在Java中写入newBigDecimal(0.1)所创建的BigDecimal正好等于 0.1(非标度值 1,其标度为 1),但是它实际上等于0.1000000000000000055511151231257827021181583404541015625。这是因为0.1无法准确地表示为 double(或者说对于该情况,不能表示为任何有限长度的二进制小数)。这样,传入到构造方法的值不会正好等于 0.1(虽然表面上等于该值)

  2. String 构造方法是完全可预知的:写入 newBigDecimal(“0.1”) 将创建一个 BigDecimal,它正好等于预期的 0.1。因此,比较而言, 通常建议优先使用String构造方法

  3. 当double必须用作BigDecimal的源时,请注意,此构造方法提供了一个准确转换;它不提供与以下操作相同的结果:先使用Double.toString(double)方法,然后使用BigDecimal(String)构造方法,将double转换为String。要获取该结果,请使用static valueOf(double)方法
    在这里插入图片描述

2.2、常用方法

  • add(BigDecimal) BigDecimal对象中的值相加,然后返回这个对象
  • subtract(BigDecimal) BigDecimal对象中的值相减,然后返回这个对象
  • multiply(BigDecimal) BigDecimal对象中的值相乘,然后返回这个对象
  • divide(BigDecimal) BigDecimal对象中的值相除,然后返回这个对象
  • toString() 将BigDecimal对象的数值转换成字符串
  • doubleValue() 将BigDecimal对象中的值以双精度数返回
  • floatValue() 将BigDecimal对象中的值以单精度数返回
  • longValue() 将BigDecimal对象中的值以长整数返回
  • intValue() 将BigDecimal对象中的值以整数返回

2.3、传统的【+、-、*、/】

public static void main(String[] args) {
    System.out.println(0.1 + 0.2);
    System.out.println(0.1 - 0.2);
    System.out.println(0.3 - 0.1);
    System.out.println(0.1 * 0.2);
    System.out.println(0.1 / 0.2);
    System.out.println(0.1 / 0.3);
}

在这里插入图片描述

结果十分的出乎意料,结果部分正确,部分不正确,这种我们没有办法预料的结果,我们在计算的时候就不能使用这些方式

出现原因

  1. 无论是 float 还是 double 都是浮点数,计算机中只有二进制,浮点数就会失去一定的精确度
  2. 十进制通常没有精确的二进制表示形式,十进制的数值二进制表示形式可能不准确,只能无限接近于精确值

2.4、BigDecimal的【+、-、*、/】

public static void main(String[] args) {
    BigDecimal one = new BigDecimal("0.3");
    BigDecimal two = new BigDecimal("0.1");
    System.out.println(one.add(two));
    System.out.println(one.subtract(two));
    System.out.println(one.multiply(two));
    System.out.println(one.divide(two));
}

在这里插入图片描述
在这里插入图片描述

  • 可以看到结果是正常的,但是在调用除法 divide() 时 idea提出警告说我们需要指定舍入
  • 还可以看到在调用除法除不尽的情况下会抛出异常,此时就涉及到我们需要指定舍入

3、BigDecimal的舍入

  • ROUND_UP :向远离0的方向舍入
  • ROUND_DOWN :向零方向舍入
  • ROUND_CEILING :向正无穷方向舍入
  • ROUND_FLOOR :向负无穷方向舍入
  • ROUND_HALF_UP :向(距离)最近的一边舍入,如果两边(的距离)是相等时,向上舍入, 1.55保留一位小数结果为1.6,也就是我们常说的“四舍五入
  • ROUND_HALF_DOWN :向(距离)最近的一边舍入,如果两边(的距离)是相等时,向下舍入, 例如1.55 保留一位小数结果为1.5
  • ROUND_HALF_EVEN :向(距离)最近的一边舍入,如果两边(的距离)是相等时,如果保留位数是奇数,使用ROUND_HALF_UP,如果是偶数,使用ROUND_HALF_DOWN
  • ROUND_UNNECESSARY :计算结果是精确的,不需要舍入模式
public static void main(String[] args) {
    BigDecimal one = new BigDecimal("0.3");
    BigDecimal two = new BigDecimal("0.1");
    System.out.println("ROUND_UP :" + two.divide(one,BigDecimal.ROUND_UP));
    System.out.println("ROUND_DOWN :" + two.divide(one,BigDecimal.ROUND_DOWN));
    System.out.println("ROUND_CEILING :" + two.divide(one,BigDecimal.ROUND_CEILING));
    System.out.println("ROUND_FLOOR :" + two.divide(one,BigDecimal.ROUND_FLOOR));
    System.out.println("ROUND_HALF_UP :" + two.divide(one,BigDecimal.ROUND_HALF_UP));
    System.out.println("ROUND_HALF_DOWN :" + two.divide(one,BigDecimal.ROUND_HALF_DOWN));
    System.out.println("ROUND_HALF_EVEN :" + two.divide(one,BigDecimal.ROUND_HALF_EVEN));
    System.out.println("ROUND_UNNECESSARY :" + two.divide(one,BigDecimal.ROUND_UNNECESSARY));
}

在这里插入图片描述

可以看到按照各种方式去取舍,但是idea依然给我们 divide() 方法爆出警告,说我们可以使用枚举常量类 RoundingMode
在这里插入图片描述

注意上面我们是没有指定小数点精确到几位的这个时候就是默认的,我们需要指定精确到小数点几位

divide() 方法有多个重载方法,可以详细看如何设置精度等问题
在这里插入图片描述

4、BigDecimal的格式化

关于Bigdecimal格式化可以使用 DecimalFormat

DecimalFormat 主要靠 # 和 0 两种占位符来指定数字长度

0:表示如果位数不足则以 0 补充

#:表示只要有可能就把数字拉上这个位置,没有数字则不补充

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/616991.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

OA系统,企业数字化转型的重要工具,用现成还是自己搭建呢

什么是OA系统 OA系统是办公自动化系统的简称,它是指一种基于计算机技术的办公工作管理系统,用于协调和规划企业内部各部门的信息发布、通信、人员流动、文档管理等方面的工作。它可以有效地提高企业办公效率和工作效益,优化企业内部沟通协作…

计算机视觉 | 深度学习预训练与MMPretrain

前言 MMPretrain是一款基于pytorch的开源深度学习预训练工具箱,是OenMMLab的项目成员之一。它是一个全新升级的预训练开源算法框架,旨在提供各种强大的预训练主干网络,并支持了不同的预训练策略。 一、MMPretrain算法库介绍 MMPretrain 源…

几分钟上线一个应用,这个神器我爱了!

配置一套公司企业运用的SaaS工作流办公管理系统需要多久?需要多少人才能开发出来?传统软件开发起码需要10个人,花上个把月时间,才能做出一套比较完整的SaaS工作流办公管理系统。 传统的开发模式它需要前后端程序员以及各平台系统的…

【Docker】浅谈Docker之AUFS、BTRFS、ZFS、Container、分层的概念

作者简介: 辭七七,目前大一,正在学习C/C,Java,Python等 作者主页: 七七的个人主页 文章收录专栏: 七七的闲谈 欢迎大家点赞 👍 收藏 ⭐ 加关注哦!💖&#x1f…

【算法】--- 几分钟了解直接选择排序(排序中最简单的排序)+快排(解决一切的优质算法)(中)

文章目录 前言🌟一、常见的排序算法:🌟二、选择排序---直接选择排序:🌏2.1.1 基本思想:🌏2.1.2 直接选择排序:🌏2.1.3 直接选择排序的特性总结:🌏2.1.4 思路&…

Vue3 Vite4 ElementPlus TS模板(含Vue-Router4+Pinia4)

引言 手动安装配置Vue3 ElementPlus模板比较繁琐,网上寻找一些模板不太符合自己预期,因此花点精力搭建一个符合自己需求的架子 采用最新的组件,版本如下: vite 4.3.9vite-plugin-mock 2.9.8vue 3.3.4pinia 2.1.3vue-router 4.2.2…

总结6种服务限流的实现方式

服务限流,是指通过控制请求的速率或次数来达到保护服务的目的,在微服务中,我们通常会将它和熔断、降级搭配在一起使用,来避免瞬时的大量请求对系统造成负荷,来达到保护服务平稳运行的目的。下面就来看一看常见的6种限流…

推荐常用的排序学习算法——BPR(贝叶斯个性化排序)

文章目录 1. 排序学习1.1 优势1.2 排序学习在推荐领域的作用1.3 排序学习设计思路1.3.1 单点法(Pointwise)1.3.2 配对法(Pairwise)1.3.3 列表法(Listwise) 2. BPR(贝叶斯个性化推荐)…

投票评选活动小程序的分享功能和背景音乐功能实现

投票评选活动小程序的分享功能和背景音乐功能实现 投票评选活动过程中,需要转发分享出去,实现投票的效果,那么就需要分享功能,不然怎么实现投票呢,其实这个是最具价值的功能之一。 而背景音乐播放功能,只…

路径规划算法:基于静电放电优化的路径规划算法- 附代码

路径规划算法:基于静电放电优化的路径规划算法- 附代码 文章目录 路径规划算法:基于静电放电优化的路径规划算法- 附代码1.算法原理1.1 环境设定1.2 约束条件1.3 适应度函数 2.算法结果3.MATLAB代码4.参考文献 摘要:本文主要介绍利用智能优化…

Qt/GUI/布局/实现窗口折叠效果/且在操作时父窗口尺寸跟随变动

文章目录 概述无法resize到小尺寸可行方案其他方案 概述 本文旨在,实现如下所示的显示或隐藏 ‘附加选项’ 的效果,以折的不常用信息和操作项,减少普通用户负担,提升用户体验。在某些软件中此类窗口折叠效果,常用 “……

SpringCloud断路器

SpringCloud断路器 Hystrix 简介 hystrix对应的中文名字是“豪猪”,豪猪周身长满了刺,能保护自己不受天敌的伤害,代表了一种防御机制。 这与hystrix本身的功能不谋而合,因此Netflix团队将该框架命名为Hystrix,并使用…

2023最详细的接口测试用例设计教程,详细文档等你来拿

目录 一、接口测试流程 二、分析接口文档元素 三、如何设计接口测试用例 四、常用的接口测试用例覆盖方法 五、接口测试接口优先级 六、接口测试的设计思路分析 七、接口测试返回结果的比较 一、接口测试流程 1、需求讨论 2、需求评审 3、场景设计 4、数据准备 5、测试执…

sdf与timingCheck和后仿真

目录 1.Distributed delays 2.specify--endspecify 1.1 specify内部语法 2.sdf 2.1 sdf的格式 3.timingCheck和网表后仿真 4.关于负值delay sdf和 module 里面的specify--endspecify都可以对路径延时进行赋值和检查;HDL语言中的‘#()’也可以描述延时【叫做D…

没事千万别动生产服数据库 - 来自小菜鸟的忠告

阿里云官方参考文档 目录 背景一、环境部署二、目录规划三、操作步骤FAQ 背景 今天把一张 5500 多万条记录的表进行按年度拆分,本来打算将表数据拆分为 2020 年、2021 年、2022 年三张新表,提升原表查询效率,仅保留 2023 年数据。表拆分完毕…

【SpinalHDL快速入门】4.1、基本类型之Bool

Tips1: 由于SpinalHDL是基于Scala构建的,Scala本身自带类似变量Boolean,故在此要认准SpinalHDL中采用的是Bool而非Boolean: Bool(大写的True和False):True表示1,False表示0Boolean&…

Vue3搭建

Vue3项目搭建全过程 vue create 项目名 选择手动吗,自定义安装 选择vue3 是否选择class风格组件 选择ts处理工具和css预处理器 Y 是否使用router的history模式 Y 选择css预处理语言 ;less 9.选择lint的检查规范 只使用EsLint官网推荐规范 使用EsLint官网推荐规…

MyBatis-plus(1)

基本概念: 一)开发效率也就是我们使用这款框架开发的速度快不快,是否简单好用易上手。从这个角度思考,每当我们需要编写一个SQL需求的时候,我们需要做几步 1)Mapper接口提供一个抽象方法 2)Mapper接口对应的映射配置文件提供对应的标签和SQL语…

论文笔记--LLaMA: Open and Efficient Foundation Language Models

论文笔记--LLaMA: Open and Efficient Foundation Language Models 1. 文章简介2. 文章概括3 文章重点技术3.1 数据集3.2 模型训练 4. 数值实验5. 文章亮点6. 原文传送门7. References 1. 文章简介 标题:LLaMA: Open and Efficient Foundation Language Models作者…

【自动化测试】--JUnit5

前言 小亭子正在努力的学习编程,接下来将开启软件测试的学习~~ 分享的文章都是学习的笔记和感悟,如有不妥之处希望大佬们批评指正~~ 同时如果本文对你有帮助的话,烦请点赞关注支持一波, 感激不尽~~ 目录 前言 Junit5简介 什么是Junit5 JU…