Java BigDecimal和double区别

转自:  https://www.cnblogs.com/mingforyou/p/3344489.html

BigDecimal类

对于不需要任何准确计算精度的数字可以直接使用float或double,但是如果需要精确计算的结果,则必须使用BigDecimal类,而且使用BigDecimal类也可以进行大数的操作。BigDecimal类的常用方法如表11-15所示。

表11-15 BigDecimal类的常用方法

序号

    法

类型

    述

1

public BigDecimal(double val)

构造

将double表示形式转换

为BigDecimal

2

public BigDecimal(int val)

构造

将int表示形式转换为

BigDecimal

3

public BigDecimal(String val)

构造

将字符串表示

形式转换为BigDecimal

4

public BigDecimal add(BigDecimal augend)

普通

加法

5

public BigDecimal subtract(BigDecimal
subtrahend)

普通

减法

6

public BigDecimal multiply(BigDecimal
multiplicand)

普通

乘法

7

public BigDecimal divide(BigDecimal
divisor)

普通

除法

范例:进行四舍五入的四则运算

复制代码
    package org.lxh.demo11.numberdemo;import java.math.BigDecimal;class MyMath {public static double add(double d1, double d2){        // 进行加法运算BigDecimal b1 = new BigDecimal(d1);BigDecimal b2 = new BigDecimal(d2);return b1.add(b2).doubleValue();}public static double sub(double d1, double d2){        // 进行减法运算BigDecimal b1 = new BigDecimal(d1);BigDecimal b2 = new BigDecimal(d2);return b1.subtract(b2).doubleValue();}public static double mul(double d1, double d2){        // 进行乘法运算BigDecimal b1 = new BigDecimal(d1);BigDecimal b2 = new BigDecimal(d2);return b1.multiply(b2).doubleValue();}public static double div(double d1,double d2,int len) {// 进行除法运算BigDecimal b1 = new BigDecimal(d1);BigDecimal b2 = new BigDecimal(d2);return b1.divide(b2,len,BigDecimal.ROUND_HALF_UP).doubleValue();}public static double round(double d,int len) {     // 进行四舍五入
    操作BigDecimal b1 = new BigDecimal(d);BigDecimal b2 = new BigDecimal(1);// 任何一个数字除以1都是原数字// ROUND_HALF_UP是BigDecimal的一个常量,
    表示进行四舍五入的操作return b1.divide(b2, len,BigDecimal.ROUND_HALF_UP).doubleValue();}}public class BigDecimalDemo01 {public static void main(String[] args) {System.out.println("加法运算:" +MyMath.round(MyMath.add(10.345,3.333), 1));System.out.println("乘法运算:" +MyMath.round(MyMath.mul(10.345,3.333), 3));System.out.println("除法运算:" +MyMath.div(10.345, 3.333, 3));System.out.println("减法运算:" +MyMath.round(MyMath.sub(10.345,3.333), 3));}}
复制代码

 

BigDecimal是Java中用来表示任意精确浮点数运算的类,在BigDecimal中,使用unscaledValue × 10-scale来表示一个浮点数。其中,unscaledValue是一个BigInteger,scale是一个int。从这个表示方法来看,BigDecimal只能标识有限小数,不过可以表示的数据范围远远大于double,在实际应用中基本足够了。

下面提一下两个精度问题:
问题一:BigDecimal的精度问题(StackOverflow上有个家伙问了相关的问题)
System.out.println(new BigDecimal(0.1).toString()); // 0.1000000000000000055511151231257827021181583404541015625
System.out.println(new BigDecimal("0.1").toString()); // 0.1
System.out.println(new BigDecimal(
Double.toString(0.1000000000000000055511151231257827021181583404541015625)).toString());// 0.1
System.out.println(new BigDecimal(Double.toString(0.1)).toString()); // 0.1

 

分析一下上面代码的问题(注释的内容表示此语句的输出)

第一行:事实上,由于二进制无法精确地表示十进制小数0.1,但是编译器读到字符串"0.1"之后,必须把它转成8个字节的double值,因此,编译器只能用一个最接近的值来代替0.1了,即0.1000000000000000055511151231257827021181583404541015625。因此,在运行时,传给BigDecimal构造函数的真正的数值是0.1000000000000000055511151231257827021181583404541015625。
第二行:BigDecimal能够正确地把字符串转化成真正精确的浮点数。
第三行:问题在于Double.toString会使用一定的精度来四舍五入double,然后再输出。会。Double.toString(0.1000000000000000055511151231257827021181583404541015625)输出的事实上是"0.1",因此生成的BigDecimal表示的数也是0.1。
第四行:基于前面的分析,事实上这一行代码等价于第三行
结论:
1.如果你希望BigDecimal能够精确地表示你希望的数值,那么一定要使用字符串来表示小数,并传递给BigDecimal的构造函数。
2.如果你使用Double.toString来把double转化字符串,然后调用BigDecimal(String),这个也是不靠谱的,它不一定按你的想法工作。
3.如果你不是很在乎是否完全精确地表示,并且使用了BigDecimal(double),那么要注意double本身的特例,double的规范本身定义了几个特殊的double值(Infinite,-Infinite,NaN),不要把这些值传给BigDecimal,否则会抛出异常。
问题二:把double强制转化成int,难道不是扔掉小数部分吗?
int x=(int)1023.99999999999999; // x=1024为什么?

原因还是在于二进制无法精确地表示某些十进制小数,因此1023.99999999999999在编译之后的double值变成了1024。

所以,把double强制转化成int确实是扔掉小数部分,但是你写在代码中的值,并不一定是编译器生成的真正的double值
验证代码:
double d = 1023.99999999999999;
int x = (int) d;
System.out.println(new BigDecimal(d).toString()); // 1024
System.out.println(Long.toHexString(Double.doubleToRawLongBits(d))); // 4090000000000000
System.out.println(x); // 1024

前面提过BigDecimal可以精确地把double表示出来还记得吧。

我们也可以直接打印出d的二进制形式,根据IEEE 754的规定,我们可以算出0x4090000000000000=(1024)。

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

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

相关文章

JDK8新特性之接口默认方法与静态方法

转载自 JDK8新特性之接口默认方法与静态方法 接口默认方法与静态方法 有这样一些场景,如果一个接口要添加一个方法,那所有的接口实现类都要去实现,而某些实现类根本就不需要实现这个方法也要写一个空实现,所以接口默认方法就是为…

mybatis generator Unknown system variable 'query_cache_size' 的解决方法

出现这种错误&#xff0c;很显然是数据库驱动程序 与 数据库版本不对应&#xff1b;如 mybatis使用 mysql-5.1.10的驱动程序&#xff0c;而mybatis配置的数据源连接的是 mysql-8.0.11 &#xff0c;修改 pom文件即可&#xff0c;如下&#xff1a; <dependency><groupId…

Java NoSuchElementException: No value present 问题解决

1 问题描述 java.util.NoSuchElementException: No value 2 问题分析 2.1 Java 1、使用stream()流里面的max().get()、min().get()、findFirst().get()方法&#xff0c;由于max()、min()、findFirst()方法会返回Optional对象&#xff0c;如果Optional对象里面没有数据&#xf…

JDK8新特性之Optional

转载自 JDK8新特性之Optional Optional是什么 java.util.Optional Jdk8提供 Optional&#xff0c;一个可以包含null值的容器对象&#xff0c;可以用来代替xx ! null的判断。 Optional常用方法 of public static <T> Optional<T> of(T value) {return new Opti…

使用maven聚合安装多个maven工程到本地仓库报错的解决方法:child module pom.xml does not exist

转自&#xff1a; https://stackoverflow.com/questions/26021141/maven-child-module-does-not-exist 1&#xff09;在maven项目Parent中的 pom.xml 中 使用聚合安装多个 maven工程到本地仓库&#xff0c;pom配置如下&#xff1a; <project xmlns"http://maven.apache…

JDK8新特性之重复注解

转载自 JDK8新特性之重复注解 什么是重复注解 下面是JDK8中的重复注解&#xff08; java.lang.annotation.Repeatable&#xff09;定义的源码。 Documented Retention(RetentionPolicy.RUNTIME) Target(ElementType.ANNOTATION_TYPE) public interface Repeatable {Class<…

JDK8之新特性扩展篇

转载自 JDK8之新特性扩展篇 BASE64 base64编码解码已经被加入到了jdk8中了。 import java.nio.charset.StandardCharsets; import java.util.Base64;public class Base64Test {public static void main(String[] args) {String text "hello javastack";String en…

eclipse maven 项目发布到tomcat 报错 Failed to scan JAR [file:/C:/xxxxx.jar] from WEB-INF/lib

报错信息如下&#xff1a; 警告: Failed to scan JAR [file:/D:/Development/Tomcat/apache-tomcat-7.0.35-64bit/webapps/Monday2/WEB-INF/lib/com.springsource.net.sf.cglib-2.2.0.jar] from WEB-INF/lib java.util.zip.ZipException: error in opening zip fileat java.uti…

SpringCloud配置中心高可用搭建

转载自 SpringCloud配置中心高可用搭建 本文通过config server连接git仓库来实现配置中心&#xff0c;除了git还可以使用svn或者系统本地目录都行。 引入依赖 <dependencies><dependency><groupId>org.springframework.cloud</groupId><artifactId…

maven项目 报错 java.lang.ClassNotFoundException: org.springframework.web.filter.HiddenHttpMethodFilter

报错信息如下&#xff1a; 严重: Exception starting filter hiddenHttpMethodFilter java.lang.ClassNotFoundException: org.springframework.web.filter.HiddenHttpMethodFilterat org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1714)at …

SpringCloud配置中心客户端读取配置

转载自 SpringCloud配置中心客户端读取配置 微服务连接配置中心来实现外部配置的读取。 引入依赖 <dependencies><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-eureka</artifactId></d…

奇技淫巧:在spring官网上下载历史版本的spring插件,springsource-tool-suite

转自&#xff1a;https://blog.csdn.net/u010203767/article/details/69211072目前spring官网(http://spring.io/tools/sts/all)上可下载的spring插件只有&#xff1a;springsource-tool-suite-3.8.4(sts-3.8.4)。但这只针对指定的eclipse版本适用。 如何为自己的eclipse下载历…

Junit5新功能一览

转载自 Junit5新功能一览Java的JUnit测试框架已经来到了5这个版本&#xff0c;与以前的版本不同的是JUnit5具有来自多个子项目的模块&#xff0c;其中包括&#xff1a; 1、Platform&#xff0c;用于在JVM上启动测试框架&#xff0c;并通过命令行定义TestEngine API。 2、Jupite…

Java多线程sleep和wait的区别,总结得非常好

转载自 Java多线程sleep和wait的区别&#xff0c;总结得非常好我们都知道sleep是让线程休眠&#xff0c;到时间后会继续执行&#xff0c;wait是等待&#xff0c;需要唤醒再继续执行&#xff0c;那么这两种方法在多线程中的表现形态&#xff0c;它们各有什么区别呢&#xff1f; …

Flex布局 让你的布局更完美

Flex布局 让你的布局更完美 CSS3 Flex布局 有很多人和我一样吧&#xff0c;刚开始学HTML的时候用的都是传统的布局&#xff0c;有时用心设计的HTMLCSS样式可能由于对方的设备、浏览器的原因&#xff0c;导致用户浏览页面时的显示效果非常丑陋。很早之前W3C为了解决解决这个问题…

mysql8.0.11 zip安装教程

转自&#xff1a; https://blog.csdn.net/hanbing5201/article/details/80101300MySQL8.0.11安装 本文总和网络上mysql安装文档总结而来。 本文安装mysql适用于1703年win10其他系统未测试 安装前准备 下载mysql安装包(. zip) 将下载的压缩包文档进行解压并将路径记录下来&#…

SpringCloud动态刷新配置信息

转载自 SpringCloud动态刷新配置信息 有时候在配置中心有些参数是需要修改的&#xff0c;这时候如何不重启而达到实时生效的效果呢&#xff1f; 添加依赖 <dependencies>...<dependency><groupId>org.springframework.boot</groupId><artifactId&g…

JavaScript 变量的作用域和生命周期

代码有生命吗&#xff1f;答案是有。它也像我们一样&#xff0c;拥有自己的生命。变量有局部变量和全局变量&#xff0c;它们的生命长短不一样。不过它的生命都是从它被声明的时候开始的&#xff0c;不同的是局部变量 会在函数运行以后被删除&#xff0c;也就是死亡&#xff0c…

MySQL笔记 - 用户管理

转自&#xff1a; https://segmentfault.com/a/1190000014856560MySQL笔记 - 用户管理 tags: 数据库 MySQL 用户管理学习目标 MySQL是一个多用户数据库&#xff0c;具有功能强大的访问控制系统&#xff0c;可以为不同用户指定允许的权限。 MySQL用户可以分为普通用户和root用户…

关于区块链技术的10本书

转载自 2018年必看&#xff1a;关于区块链技术的10本书 1、区块链技术指南 区块链专家联袂推荐&#xff0c;资深区块链践行者联合撰写&#xff0c;从技术层面全面揭示区块链技术秘密。涵盖基础概念、架构、底层算法、应用开发、典型的区块链解决方案、常见问题等读者*为关心的技…