Java使用BigDecimal(公式精确计算)+(精度丢失问题)

news2024/9/29 9:30:44

一、Java使用BigDecimal公式计算(精确计算)

介绍:

       使用BigDecimal加减乘除方法运算,可以使用BigDecimal类提供的add、subtract、multiply、divide方法函数实现。

公式加法计算~add

  public  static void main(String[] args){

       BigDecimal a = BigDecimal.valueOf(5.6);

       BigDecimal b = BigDecimal.valueOf(2.1);

       //BigDecimal计算add
       BigDecimal addResult = a.add(b);
       System.out.println("结果集: " +addResult);

       
   }

结果集: 

公式减法计算~subtract

public  static void main(String[] args){

       BigDecimal a = BigDecimal.valueOf(5.6);

       BigDecimal b = BigDecimal.valueOf(2.1);

       BigDecimal计算subtract
       BigDecimal subtract = a.subtract(b);

       System.out.println("结果集subtract: " +subtract);


   }

结果集:

 

公式乘法计算~multiply

public  static void main(String[] args){

       BigDecimal a = BigDecimal.valueOf(5.6);

       BigDecimal b = BigDecimal.valueOf(2.1);

       BigDecimal计算multiply
       BigDecimal multiply = a.multiply(b);

       System.out.println("结果集: " +multiply);


   }

结果集:

 

公式除法计算~divide

public  static void main(String[] args){

       BigDecimal a = BigDecimal.valueOf(5.6);

       BigDecimal b = BigDecimal.valueOf(2.1);

       //BigDecimal计算divide
       //ROUND_HALF_UP:向“最接近的”整数舍入。 若舍入位大于等于5,则对舍入部分的前一位数字加1;若舍入位小于5,则直接舍弃。即为四舍五入模式。
       BigDecimal divide = a.divide(b,2,BigDecimal.ROUND_HALF_UP);//四舍五入,保留两位小数.

       System.out.println("结果集: " +divide);


   }

结果集:

二、 BigDecimal(舍入模式)选择

简介:

BigDecimal.setScale主要用于对BigDecimal数据小数点后的位数进行 进位、舍位、截断等操作

java.math.RoundingMode:这是一种枚举类型,它定义了8种数据的舍入模式。它与java.math.BigDecimal类中定义的8个同名静态常量的作用相同,可用BigDecimal.setScale(int newScale, RoundingMode roundingMode)来设置数据的精度和舍入模式。
 

1、ROUND_UP:向远离零的方向舍入。

        若舍入位为非零,则对舍入部分的前一位数字加1;若舍入位为零,则直接舍弃。即为向外取整模式。

2、ROUND_DOWN:向接近零的方向舍入。

        不论舍入位是否为零,都直接舍弃。即为向内取整模式。

3、ROUND_CEILING:向正无穷大的方向舍入。

        若 BigDecimal 为正,则舍入行为与 ROUND_UP 相同;若为负,则舍入行为与 ROUND_DOWN 相同。即为向上取整模式。

4、ROUND_FLOOR:向负无穷大的方向舍入。

        若 BigDecimal 为正,则舍入行为与 ROUND_DOWN 相同;若为负,则舍入行为与 ROUND_UP 相同。即为向下取整模式。

5、ROUND_HALF_UP:向“最接近的”整数舍入。

        若舍入位大于等于5,则对舍入部分的前一位数字加1;若舍入位小于5,则直接舍弃。即为四舍五入模式。

6、ROUND_HALF_DOWN:向“最接近的”整数舍入。

        若舍入位大于5,则对舍入部分的前一位数字加1;若舍入位小于等于5,则直接舍弃。即为五舍六入模式。

7、ROUND_HALF_EVEN:向“最接近的”整数舍入。

        若(舍入位大于5)或者(舍入位等于5且前一位为奇数),则对舍入部分的前一位数字加1;

        若(舍入位小于5)或者(舍入位等于5且前一位为偶数),则直接舍弃。即为银行家舍入模式。

8、ROUND_UNNECESSARY

        断言请求的操作具有精确的结果,因此不需要舍入。

        如果对获得精确结果的操作指定此舍入模式,则抛出ArithmeticException。
 

案例: 

 public  static void main(String[] args){

       BigDecimal a = BigDecimal.valueOf(5.6);

       BigDecimal b = BigDecimal.valueOf(2.1);

       //BigDecimal计算divide
       //ROUND_HALF_UP:向“最接近的”整数舍入。 若舍入位大于等于5,则对舍入部分的前一位数字加1;若舍入位小于5,则直接舍弃。即为四舍五入模式。
       //ROUND_UP:向远离零的方向舍入。  若舍入位为非零,则对舍入部分的前一位数字加1;若舍入位为零,则直接舍弃。即为向外取整模式。
       BigDecimal divide = a.divide(b,BigDecimal.ROUND_HALF_UP, 2).setScale(2,ROUND_UP);

       System.out.println("结果集: " +divide);


   }

结果集:

计算器结果集如下图:

三、 BigDecimal.setScale用法案例

BigDecimal.setScale主要用于对BigDecimal数据小数点后的位数进行 进位、舍位、截断等操作

public  static void main(String[] args){

       BigDecimal a = BigDecimal.valueOf(5.6);

       BigDecimal b = BigDecimal.valueOf(2.1);

       //BigDecimal计算divide
       BigDecimal divide = a.divide(b,2,BigDecimal.ROUND_UP).setScale(2, BigDecimal.ROUND_UP);

       System.out.println("结果集: " +divide);


   }

结果集:

四、Java使用BigDecimal精度丢失问题

介绍:

在实际应用中,经常会使用到计算,举例如 '银行' 特别是在我们交易的时候,计算结果要求的更加精确,这个时候我们就会使用到java.math包中提供的API类BigDecimal,用于对超过16位有效位数的数字进行精确的计算。

以如下代码为例:

public  static void main(String[] args){

       System.out.println("使用bigDecimal进行转换"+new BigDecimal(234.567));

   }

运行后结果集出现精度丢失: 

结果可以看出使用new BigDecimal(234.567),传入参数为double类型发生精度丢失,运行的结果并不等于234.567,而是为234.5670000000000072759576141834259033203125。

原因是double不能表示为任何有限长度的二进制小数;

解决方法:     使用new BigDecimal(String)构造函数,创建一个参数以字符串表示数值的对象
 

public  static void main(String[] args){

       double a = 234.567;

       String  b = "234.567";

       // 方式一:double的封装类Double中的toString()方法解决
       System.out.println("使用bigDecimal进行转换" + new BigDecimal(new Double(a).toString()));

       // 方式二:使用String接收数值
       System.out.println("使用bigDecimal进行转换" + new BigDecimal(b));


   }

 结果集如下:

 如需使用加减乘除方法运算,可以使用BigDecimal类提供的add、subtract、multiply、divide方法实现 如上:Java使用BigDecimal公式计算

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

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

相关文章

动态规划合集

62 斐波那契数列 public class Solution {public int Fibonacci(int n) { return f(n);}public int f(int n){if(n1||n2){return 1;}return f(n-1)f(n-2);} }这种做法时间复杂度O(2^N),空间复杂度是用递归栈,O(n) 改进:用动态规划,可以…

Pytorch优化器全总结(三)牛顿法、BFGS、L-BFGS 含代码

目录 写在前面 一、牛顿法 1.看图理解牛顿法 2.公式推导-三角函数 3.公式推导-二阶泰勒展开 二、BFGS公式推导 三、L-BFGS 四、算法迭代过程 五、代码实现 1.torch.optim.LBFGS说明 2.使用LBFGS优化模型 优化器系列文章列表 Pytorch优化器全总结(一&…

C 程序设计教程(09)—— 数据输出函数(printf)用法详解

C 程序设计教程(09)—— 数据输出函数(printf)用法详解 该专栏主要介绍 C 语言的基本语法,作为《程序设计语言》课程的课件与参考资料,用于《程序设计语言》课程的教学,供入门级用户阅读。 目录…

Python小案例

1、简单的打印输出 age =18 print("我的名字是%s,我的国籍是%s"%("小张","中国")) print("我的年纪是:%d岁"%age) print("www","baidu","com",sep=".") #sep是使用.分割的意思,这个输出是百…

微信小程序开发——小程序的宿主环境API,协同工作和发布

一.小程序API概述 小程序中的 API 是由宿主环境提供的,通过这些丰富的小程序 API ,开发者可以方便的调用微信提供的能力,例如:获取用户信息、本地存储、支付功能等。 二.小程序API的3大分类 a.事件监听AP1 特点:以…

【服务器数据恢复】服务器硬盘掉线的数据库数据恢复案例

服务器数据恢复环境&故障: 某公司服务器,配备24块FC硬盘,两块硬盘出现故障掉线,导致服务器上层的卷无法挂载。 服务器数据恢复过程: 1、查看服务器硬盘状态发现有两块硬盘离线,将服务器内的所有硬盘做好…

【数据结构-JAVA】栈(Stack)和队列(Queue)

栈1.1 栈的概念栈:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈 顶,另一端称为栈底。栈中的数据元素遵守先进后出,后进先出的原则(LIFO——Last In First Out&a…

【从零开始学习深度学习】41. 算法优化之RMSProp算法【基于AdaGrad算法的改进】介绍及其Pytorch实现

上一篇文章AdaGrad算法中提到,因为调整学习率时分母上的变量st\boldsymbol{s}_tst​一直在累加按元素平方的小批量随机梯度,所以目标函数自变量每个元素的学习率在迭代过程中一直在降低(或不变)。因此,当学习率在迭代早…

LeetCode 45. 跳跃游戏 II

45. 跳跃游戏 II - 力扣(LeetCode) 解法1:(动态规划 贪心) 果然代码越短,思路越难。这题用的是动态规划贪心的思想。首先分析题意我们可以知道,从索引0这个点开始,我们走一步可以…

redis命令第二弹

1、redis命令-hash类型练习2、redis命令-list类型练习3、redis命令-set类型练习

YOLOV5环境搭建以及训练COCO128数据集

前言记录了自己训练coco128的全过程手把手教你YOLOV5环境搭建以及训练COCO128数据集。相关配置文件在百度网盘中。如果懒得话可以直接全部用我的数据一、准备工作1.1创建环境打开anaconda power shell(最好以管理员身份运行,免得到后面相关文件权限进不去…

sentinel-介绍(一)

Sentinel Website(Sentinel 官网网站) Sentinel: 分布式系统的流量防卫兵 Sentinel 是什么? 随着微服务的流行,服务和服务之间的稳定性变得越来越重要。Sentinel 以流量为切入点,从流量控制、流量路由、熔断降级、系…

ansible配置yum源仓库

1.挂载本地光盘到/mnt 2.配置yum源仓库文件通过多种方式实现 仓库1 : Name: RH294_Base Description: RH294 base software Base urt: file:///mnt/BaseOS 不需要验证钦件包 GPG 签名 启用此软件仓库 仓库 2: Name: RH294_S…

LeetCode刷题模版:41 - 50

目录 简介41. 缺失的第一个正数42. 接雨水43. 字符串相乘44. 通配符匹配45. 跳跃游戏 II46. 全排列47. 全排列 II48. 旋转图像49. 字母异位词分组50. Pow(x, n)结语简介 Hello! 非常感谢您阅读海轰的文章,倘若文中有错误的地方,欢迎您指出~ ଘ(੭ˊᵕˋ)੭ 昵称:海轰 标…

axios系列之取消请求

文章の目录写在最后使用 cancel token 取消请求 Axios 的 cancel token API 基于cancelable promises proposal,它还处于第一阶段。 可以使用 CancelToken.source 工厂方法创建 cancel token,像这样: const CancelToken axios.CancelToken;…

Revit二次开发小技巧(十七)实时监控模型线的生成

前言:项目中需要一个需求,用户想调用出Revit中自带的绘制模型线方法,然后再绘制结束时,可以拿到绘制的模型线,然后实现后面的算法。这里记录一种方法,通过DocumentChange事件修改Tag的PropertyChanged事件来…

【Python】pandas获取全省人口数据并作可视化分析

前言 今天我们看看自己所在的省份的人口人数,使用pandas并作可视化分析。 环境使用 python 3.9pycharm 模块使用 pandasPandas 是基于NumPy的一种工具,该工具是为解决数据分析任务而创建的。Pandas 纳入了大量库和一些标准的数据模型,提供…

java和vue募捐网水滴筹项目捐款爱心系统筹款系统

简介 募捐网,注册用户实名认证通过后可以发布募捐,管理员审核募捐通过后,前台用户可以看到该募捐信息,进行募捐或者举报(管理审核举报成功后,会拉黑该募捐发起人),前台展示公告、爱…

83. 删除排序链表中的重复元素(链表)

文章目录题目描述方法一 暴力法方法二 递归法参考文献题目描述 给定一个已排序的链表的头 head , 删除所有重复的元素,使每个元素只出现一次 。返回 已排序的链表 。 示例 1: 输入:head [1,1,2] 输出:[1,2] 示例 2…

酷开系统——家庭场景下的智能营销系统!

随着人们生活方式的改变,以往传统的营销资源和渠道正在慢慢陷入一个“无用”的尴尬境地,而作为家庭娱乐中心的智能大屏,近两年所表现出来的数据和效果却逐渐备受企业和品牌方关注,有数据显示,智能大屏的家庭覆盖规模正…