当前位置: 首页 > news >正文

BigDecimal高精度运算

1. BigDecimal是什么类型,为什么可以转为double

BigDecimal 是 Java 中用于表示任意精度的十进制数的类。它主要用于金融和商业计算,能够提供比 double 类型更高精度的运算,特别是在处理货币等需要精确计算的场景中。

1.1  BigDecimal 的基本性质

  • 高精度: BigDecimal 可以表示非常大或者非常小的十进制数,并且能够精确控制小数点后的位数。
  • 不可变: BigDecimal 类是不可变的,这意味着每次对 BigDecimal 对象进行操作时,都会返回一个新的 BigDecimal 对象,而不会改变原对象。
  • 构造方法: BigDecimal 可以通过字符串、doubleint 等多种方式来构造。例如:
BigDecimal bd1 = new BigDecimal("123.45"); // 通过字符串构造
BigDecimal bd2 = new BigDecimal(123.45);   // 通过double构造(不推荐,因为可能会导致精度问题)

1.2. 为什么可以转换为 double

BigDecimal 可以通过调用其 doubleValue() 方法将其转换为 double 类型。这是因为 BigDecimal 本质上是一个表示十进制数的对象,而 double 是 Java 的原始类型,用于表示双精度浮点数。

BigDecimal bd = new BigDecimal("123.45");
double d = bd.doubleValue(); // 将 BigDecimal 转换为 double

1.3. 转换的注意事项

虽然可以将 BigDecimal 转换为 double,但需要注意以下几点:

  • 精度丢失: double 的精度有限(大约 15-16 位有效数字),而 BigDecimal 可以精确到任意位数。因此,将 BigDecimal 转换为 double 时,可能会出现精度丢失的问题,特别是在处理非常大的数字或精度要求很高的场景时。
  • 浮点运算误差: 使用 double 进行运算时,可能会引入浮点运算误差,而 BigDecimal 在执行加减乘除等操作时可以保持精确。

1.4. 使用 doubleValue() 的场景

在某些情况下,尽管 BigDecimal 提供了更高精度,但仍需要将其转换为 double 类型,例如:

  • 与其他使用 double 的库或 API 进行交互:某些库或 API 可能要求使用 double 类型。
  • 计算后输出结果:在计算完成后,可能需要将结果输出为 double 类型以便于显示或进一步处理。

总之,BigDecimal 是一种用于高精度计算的数据类型,可以通过 doubleValue() 方法转换为 double,但在转换过程中应小心处理精度丢失和浮点运算误差的可能性。

2. 两个整数相除为什么能得到BigDecimal 类型

在 Java 中,两个整数相除通常会得到一个整数结果,但如果你希望得到一个高精度的结果并且避免精度丢失,可以使用 BigDecimal 类来进行除法运算。通过 BigDecimal 的除法操作,即使两个整数相除,结果也会保留小数部分,从而得到一个 BigDecimal 类型的结果。

2.1. NumberUtil.div 方法解析

NumberUtil.div 方法来自于 Hutool 工具包,是一个方便的工具方法,用于高精度的除法运算。其主要作用是将两个数(可以是整数、小数等)进行除法运算,并返回 BigDecimal 类型的结果。

例如,假设 volumecarriage.getLightThrowingCoefficient() 都是整数类型(例如 intlong),但通过 NumberUtil.div 进行除法运算时,会返回一个精确的 BigDecimal 结果,而不是简单地返回整数部分。

2.2. Hutool 的 NumberUtil.div 用法

NumberUtil.div 的用法通常如下:

public static BigDecimal div(Number v1, Number v2, int scale)
  • v1:被除数,可以是整数、浮点数、BigDecimal 等类型。
  • v2:除数,可以是整数、浮点数、BigDecimal 等类型。
  • scale:指定结果的小数位数,即结果精度。

2.3. 举例说明

假设 volume 是 10,carriage.getLightThrowingCoefficient() 是 3,你想要精确地计算这两个数的商并保留一位小数。通过 NumberUtil.div 进行除法运算会返回一个 BigDecimal 类型的结果。

int volume = 10;
int coefficient = 3;
BigDecimal volumeWeight = NumberUtil.div(volume, coefficient, 1);
  • 这个除法计算实际上是 10 / 3,结果是 3.3333...,但由于你指定了保留一位小数(scale 参数为 1),结果会被四舍五入为 3.3,并以 BigDecimal 类型返回。

2.4. BigDecimal 的除法

BigDecimal 本身也提供了非常强大的除法操作。如果不使用 NumberUtil.div,也可以直接使用 BigDecimal 进行计算:

BigDecimal volume = new BigDecimal(10);
BigDecimal coefficient = new BigDecimal(3);
BigDecimal volumeWeight = volume.divide(coefficient, 1, RoundingMode.HALF_UP);
  • divide 方法BigDecimaldivide 方法提供了高精度的除法运算。
  • 1 表示结果保留一位小数。
  • RoundingMode.HALF_UP 表示四舍五入。

2.5. 总结

当两个整数进行除法运算时,直接相除会丢弃小数部分,得到一个整数结果。然而,通过使用 BigDecimal 类或工具类方法(如 Hutool 的 NumberUtil.div),可以保留小数部分并得到一个精确的 BigDecimal 结果。这种方法特别适用于需要高精度计算的场景,例如财务、科学计算等。

3. 代码解释

/**************************
* 参数类型说明:
* volumes是Integer类型
* carriage.getLightThrowingCoefficient()是Integer类型
* waybillDTO.getWeight() 是double类型
**************************/// 计算体积重量,体积 / 轻抛系数
BigDecimal volumeWeight = NumberUtil.div(volume, carriage.getLightThrowingCoefficient(), 1);//取大值
double computeWeight = NumberUtil.max(volumeWeight.doubleValue(), NumberUtil.round(waybillDTO.getWeight(), 1).doubleValue());

这段代码的目的是根据包裹的体积和重量计算出实际的计费重量。它首先计算体积重量,然后在体积重量和实际重量之间取较大值作为最终的计费重量。

3.1 计算体积重量

BigDecimal volumeWeight = NumberUtil.div(volume, carriage.getLightThrowingCoefficient(), 1);
  • volumeWeight 是通过将包裹的体积除以一个称为“轻抛系数”的值计算出来的。
    • volume: 包裹的体积(通常是长、宽、高的乘积)。
    • carriage.getLightThrowingCoefficient(): 轻抛系数,这是一个常量,用来将体积转换为重量。通常,物流行业使用这个系数来计算体积重量,因为轻的、但体积大的包裹会占用更多的运输空间。
    • NumberUtil.div: 用于进行高精度的除法运算。
      • 第一个参数 volume 是被除数(体积)。
      • 第二个参数 carriage.getLightThrowingCoefficient() 是除数(轻抛系数)。
      • 第三个参数 1 表示结果保留一位小数。

体积重量的计算公式是: 体积重量=包裹体积/轻抛系数

3.2. 取较大值作为计费重量

double computeWeight = NumberUtil.max(volumeWeight.doubleValue(), NumberUtil.round(waybillDTO.getWeight(), 1).doubleValue());

computeWeight 计算包裹的最终计费重量,它取的是体积重量和实际重量中的较大值。

  • volumeWeight.doubleValue(): 将体积重量转换为 double 类型,以便与实际重量比较。
  • waybillDTO.getWeight(): 获取包裹的实际重量。
  • NumberUtil.round(waybillDTO.getWeight(), 1): 对实际重量进行四舍五入,保留一位小数。
  • NumberUtil.max: 比较两个数值,返回较大者。

注:NumberUtil.round 方法的作用是对数字进行四舍五入,并返回一个 BigDecimal 类型的结果。即使 waybillDTO.getWeight() 返回的是 double 类型,经过 NumberUtil.round 处理后,结果会变成 BigDecimal

相关文章:

BigDecimal高精度运算

1. BigDecimal是什么类型,为什么可以转为double BigDecimal 是 Java 中用于表示任意精度的十进制数的类。它主要用于金融和商业计算,能够提供比 double 类型更高精度的运算,特别是在处理货币等需要精确计算的场景中。 1.1 BigDecimal 的基…...

C/C++实现蓝屏2.0

🚀欢迎互三👉:程序猿方梓燚 💎💎 🚀关注博主,后期持续更新系列文章 🚀如果有错误感谢请大家批评指出,及时修改 🚀感谢大家点赞👍收藏⭐评论✍ 前…...

Unity音频管理器插件AudioToolKit

Unity音频管理器插件AudioToolKit 介绍AudioToolKit介绍具体用法总结 介绍 最近在自己写音频管理器的时候在网上发现了一款比较好用并且功能很全的一个音频管理插件,叫做AudioToolKit的插件。 如果需要的可以直接从我资源中找AudioToolKit。 AudioToolKit介绍 A…...

搜维尔科技:驾驶模拟器背后的技术: Varjo的虚拟/混合现实 (VR/XR)提供独特的优势,最终加快汽车开发创新的步伐

专业驾驶模拟器广泛应用于车辆开发,帮助汽车行业在开发过程的早期做出更好的设计决策。总体目标是为测试驾驶员提供最真实的驾驶体验,包括动态动作和声音,并测试控制算法或辅助系统等功能。环境越真实,驾驶员的体验就越接近最终车…...

OSL 冠名赞助Web3峰会 “FORESIGHT2024”圆满收官

OSL 望为香港数字资产市场发展建设添砖加瓦 (香港,2024 年 8 月 13 日)- 8 月 11 日至 12 日, 由 香港唯一专注数字资产的上市公司 OSL 集团(863.HK)冠名赞助,Foresight News、 Foresight Ventu…...

LeetCode 3148.矩阵中的最大得分:每个元素与其左或上元素之差的最大值(原地修改O(1)空间)

【LetMeFly】3148.矩阵中的最大得分:每个元素与其左或上元素之差的最大值(原地修改O(1)空间) 力扣题目链接:https://leetcode.cn/problems/maximum-difference-score-in-a-grid/ 给你一个由 正整数 组成、大小为 m x n 的矩阵 g…...

主流的开源大型语言模型

本期我们来聊聊目前主流的开源大型语言模型。这些模型就像是AI界的超级英雄,各具特色,为我们的研究和开发提供了强大的力量。🚀 GPT-Neo:这是EleutherAI的杰作,它模仿了OpenAI的GPT-3。GPT-Neo虽然规模小一些&#xf…...

【自动驾驶】话题通信

目录 构建发布者构建订阅者编写lanch文件自动启动节点测试运行ROS的目录结构 切换到工作空间的src目录下: 构建发布者 catkin_create_pkg publisher std_msgs rospy roscpp编写发布者程序: // 1.包含头文件 #include "ros/ros.h" #include &…...

【Linux】中的软件安装:深入探索RPM、SRPM与YUM

🐇明明跟你说过:个人主页 🏅个人专栏:《Linux :从菜鸟到飞鸟的逆袭》🏅 🔖行路有良友,便是天堂🔖 目录 一、引言 1、Linux的起源与发展 2、RPM、SRPM与YUM的简要介…...

uniapp自定义请求头信息header

添加请求头:uniapp自定义请求头信息header,如下:添加tenant-id参数 代码...

SpringBoot整合Liquibase

1、是什么? Liquibase官网 Liquibase是一个开源的数据库管理工具,可以帮助开发人员管理和跟踪数据库变更。它可以与各种关系型数据库和NoSQL数据库一起使用,并提供多种数据库任务自动化功能,例如数据库迁移、版本控制和监控。Li…...

虚幻5|给武器添加碰撞检测与伤害

本章内容衔接上两章,需要完成上两章才能用本章内容 虚幻5|角色武器装备的数据库学习(不只是用来装备武器,甚至是角色切换也很可能用到)-CSDN博客虚幻5|普通攻击,使用接口更方便-CSDN博客 如有疑问,可访问…...

RESTful API设计指南:构建高效、可扩展的Web服务

目录 引言 一.RESTful API概述 二.设计原则 2.1. 资源导向 2.2. 使用标准的HTTP方法 2.3. 无状态通信 2.4. 可缓存响应 2.5. 分层系统 2.6. 按需加载代码(可选) 2.7. HATEOAS 三.最佳实践 3.1. 明确资源和子资源 3.2. 使用合适的HTTP状态码 …...

黑马头条vue2.0项目实战(九)——编辑用户资料

目录 1. 创建组件并配置路由 2. 页面布局 3. 展示用户信息 4. 修改昵称 5. 修改性别 6. 修改生日 7. 修改头像 7.1 图片上传预览 7.2 使用纯客户端的方式处理用户头像上传预览 7.3 头像裁切 7.4 纯客户端的图片裁切上传流程 7.5 Cropper.js 图片裁剪器的基本使用 …...

43.【C语言】指针(重难点)(F)

目录 15.二级指针 *定义 *演示 16.三级以及多级指针 *三级指针的定义 *多级指针的定义 17.指针数组 *定义 *代码 18.指针数组模拟二维数组 往期推荐 15.二级指针 *定义 之前讲的指针全是一级指针 int a 1; int *pa &a;//一级指针 如果写成 int a 1; int *pa &a…...

【STM32+HAL】杆球控制系统

一、前言 2017年电赛出了道板球控制系统题目,现写一个简化版本——杆球控制系统,以此记录电赛集训生活。 二、题目分析 最终采取的方案是:OpenMV读取小球的当前位置,并将坐标值传给STM32端,再由32通过电机改变杆的位置…...

用Python实现9大回归算法详解——04. 多项式回归算法

多项式回归 是线性回归的一种扩展,它通过将输入特征的多项式项(如平方、立方等)引入模型中,以捕捉数据中非线性的关系。虽然多项式回归属于线性模型的范畴,但它通过增加特征的多项式形式,使得模型能够拟合非…...

vue打包更新packge.json版本号

VUE项目打包自动更新版本号 此方法只针对 Vue 如果使用其他框架,可以此参照作为参考 一、先看效果 二、创建 buildVersion.js 文件 文件内容 目前只针对3位版本号 递增规则是 每次更新 加一次小版本,10次小版本向前递增一个版本。如:1.0.9 递…...

计算机视觉技术解析:从基础到前沿

第一部分:计算机视觉基础与基本原理 计算机视觉是人工智能领域的一个重要分支,旨在使计算机能够理解和处理图像和视频数据。随着深度学习技术的飞速发展,计算机视觉已经在许多实际应用场景中取得了显著的成果,如图像识别、目标检…...

unity游戏开发003:深入理解Unity中的坐标系

Unity游戏开发 “好读书,不求甚解;每有会意,便欣然忘食。” 本文目录: Unity游戏开发 Unity游戏开发深入理解Unity中的坐标系前言1. 坐标轴2. 左手坐标系3. 世界坐标系 vs. 局部坐标系4. 坐标变换5. 注意事项 总结 深入理解Unity中…...

基于FPGA的PID算法学习———实现PID比例控制算法

基于FPGA的PID算法学习 前言一、PID算法分析二、PID仿真分析1. PID代码2.PI代码3.P代码4.顶层5.测试文件6.仿真波形 总结 前言 学习内容:参考网站: PID算法控制 PID即:Proportional(比例)、Integral(积分&…...

阿里云ACP云计算备考笔记 (5)——弹性伸缩

目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...

MongoDB学习和应用(高效的非关系型数据库)

一丶 MongoDB简介 对于社交类软件的功能,我们需要对它的功能特点进行分析: 数据量会随着用户数增大而增大读多写少价值较低非好友看不到其动态信息地理位置的查询… 针对以上特点进行分析各大存储工具: mysql:关系型数据库&am…...

ESP32读取DHT11温湿度数据

芯片:ESP32 环境:Arduino 一、安装DHT11传感器库 红框的库,别安装错了 二、代码 注意,DATA口要连接在D15上 #include "DHT.h" // 包含DHT库#define DHTPIN 15 // 定义DHT11数据引脚连接到ESP32的GPIO15 #define D…...

spring:实例工厂方法获取bean

spring处理使用静态工厂方法获取bean实例,也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下: 定义实例工厂类(Java代码),定义实例工厂(xml),定义调用实例工厂&#xff…...

新能源汽车智慧充电桩管理方案:新能源充电桩散热问题及消防安全监管方案

随着新能源汽车的快速普及,充电桩作为核心配套设施,其安全性与可靠性备受关注。然而,在高温、高负荷运行环境下,充电桩的散热问题与消防安全隐患日益凸显,成为制约行业发展的关键瓶颈。 如何通过智慧化管理手段优化散…...

学习STC51单片机32(芯片为STC89C52RCRC)OLED显示屏2

每日一言 今天的每一份坚持,都是在为未来积攒底气。 案例:OLED显示一个A 这边观察到一个点,怎么雪花了就是都是乱七八糟的占满了屏幕。。 解释 : 如果代码里信号切换太快(比如 SDA 刚变,SCL 立刻变&#…...

分布式增量爬虫实现方案

之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面,避免重复抓取,以节省资源和时间。 在分布式环境下,增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路:将增量判…...

AI病理诊断七剑下天山,医疗未来触手可及

一、病理诊断困局:刀尖上的医学艺术 1.1 金标准背后的隐痛 病理诊断被誉为"诊断的诊断",医生需通过显微镜观察组织切片,在细胞迷宫中捕捉癌变信号。某省病理质控报告显示,基层医院误诊率达12%-15%,专家会诊…...

JavaScript基础-API 和 Web API

在学习JavaScript的过程中,理解API(应用程序接口)和Web API的概念及其应用是非常重要的。这些工具极大地扩展了JavaScript的功能,使得开发者能够创建出功能丰富、交互性强的Web应用程序。本文将深入探讨JavaScript中的API与Web AP…...