精确值避免使用float和double,使用BigDecimal
实现将double类型的值转换为BigDecimal类型的值的不同途径以及各途径间的区别
一:有人可能认为在 Java 中写入 new BigDecimal(0.1) 所创建的 BigDecimal 正好等于 0.1(非标度值 1,其标度为 1),
但是它实际上等于 0.1000000000000000055511151231257827021181583404541015625。
这是因为 0.1 无法准确地表示为 double(或者说对于该情况,不能表示为任何有限长度的二进制小数)。
这样,传入 到构造方法的值不会正好等于 0.1(虽然表面上等于该值)。
二:另一方面,String 构造方法是完全可预知的:写入 new BigDecimal("0.1") 将创建一个 BigDecimal,
它正好 等于预期的 0.1。因此,比较而言,通常建议优先使用 String 构造方法。
三:使用public static BigDecimal valueOf(double val)
使用 Double.toString(double) 方法提供的 double 规范的字符串表示形式将 double 转换为 BigDecimal。
这通常是将 double(或 float)转化为 BigDecimal 的首选方法, 因为返回的值等于从构造 BigDecimal(使用 Double.toString(double) 得到的结果)得到的值。
package com; import java.math.BigDecimal; /**
* Created by huyanxia on 2017/11/27.
*/
public class TestBigDecimal {
public static void main(String[] args) {
//3.0300000000000002
System.out.println(1.03 + 2);
//3.0300000000000000266453525910037569701671600341796875
System.out.println(new BigDecimal(1.03).add( new BigDecimal(2)));
//3.03
System.out.println(new BigDecimal("1.03").add( new BigDecimal("2")));
//3.03
System.out.println(BigDecimal.valueOf(new BigDecimal("1.03").add( new BigDecimal("2")).doubleValue()));
/* float和double尤其不适合进行货币计算,原因在于我们的计算机是二进制的。浮点数没有办法是用二进制进行精确表示。
我们的CPU表示浮点数由两个部分组成:指数和尾数,这样的表示方法一般都会失去一定的精确度,有些浮点数运算也会产生一定的误差。
如:2.4的二进制表示并非就是精确的2.4。反而最为接近的二进制表示是 2.3999999999999999。浮点数的值实际上是由一个特定的数学公式计算得到的。*/ /*解决: 使用BigDecimal、int或long进行货币计算,1.BigDecimal货币计算,BigDecimal的参数要以字符串而不是double
2.以分为单位进行计算,而不是意愿为单位*/ //可以买3个,还剩0.3999999999999999元。
double funds = 1.00;
int bought = 0;
for(double price = 0.1; funds >= price; price +=0.1){
funds -= price;
bought ++;
}
System.out.println("可以买" + bought + "个,还剩" + funds +"元。"); //可以买4个,还剩0.00元。
BigDecimal TEN_CEN = new BigDecimal("0.1");
BigDecimal funds1 = new BigDecimal("1.00");
int bought1 = 0;
for(BigDecimal price = TEN_CEN; funds1.compareTo(price) >= 0; price = price.add(TEN_CEN)){
funds1 = funds1.subtract(price);
bought1 ++;
}
System.out.println("可以买" + bought1 + "个,还剩" + funds1 +"元。"); //可以买3个,还剩0.3999999999999999666933092612453037872910499572753906250元。
BigDecimal TEN_CEN2 = new BigDecimal(0.1);
BigDecimal funds2 = new BigDecimal(1.00);
int bought2 = 0;
for(BigDecimal price = TEN_CEN2; funds2.compareTo(price) >= 0; price = price.add(TEN_CEN2)){
funds2 = funds2.subtract(price);
bought2 ++;
}
System.out.println("可以买" + bought2 + "个,还剩" + funds2 +"元。"); //可以买4个,还剩0元。
int funds3 = 100;
int bought3 = 0;
for(int price = 10; funds3 >= price; price += 10){
funds3 -= price;
bought3 ++;
}
System.out.println("可以买" + bought3 + "个,还剩" + funds3 +"元。");
}
}
精确值避免使用float和double,使用BigDecimal的更多相关文章
- Char、float、Double、BigDecimal
Char初识 char: char类型是一个单一的 16 位 Unicode 字符 char 在java中是2个字节("字节"是byte,"位"是bit ,1 ...
- float、double、BigDecimal的一些精度问题
float f = 280.8f;System.out.println(f*100);结果是什么?结果是:28080.0f(我是这么想的)实际结果是:28079.998 既然float处理有问题换do ...
- 第48条:如果需要精确的答案,请避免使用float和double
float和double主要为了科学计算和工程计算而设计,执行二进制浮点运算,这是为了在广泛的数值范围上提供较为精确的快速近似计算而精心设计的.然而,它们没有提供完全精确的结果,所以不适合用于需要精确 ...
- decimal, float 和double
阿里的 Java 手册里写着: 6. [强制] 小数类型为 decimal,禁止使用 float 和 double. 说明:float 和 double 在存储的时候,存在精度损失的问题,很可能在值的 ...
- 关于 BigDecimal处理float、double数据
Big Decimal 在java中,对于float与double中的数据,总会因为精度问题而丢失数据的准确性,也就是说对于两者所处理的得到的值是无限接近于那个数,而并非一个精确数字,而对于电商中所涉 ...
- float和double精度问题
System.out.println(new BigDecimal(253.90).doubleValue() * 100);25390.0精度正确 System.out.println(new Bi ...
- Effective Java 48 Avoid float and double if exact answers are required
Reason The float and double types are particularly ill-suited for monetary calculations because it i ...
- Decimal、 Float、 Double 使用
一.Java 1.float型定义的数据末尾必须 有"f "或"F",为了和double区别.例float x=123.456f, y=2e20f; publ ...
- java float、double精度研究(转)
在java中运行一下代码System.out.println(2.00-1.10);输出的结果是:0.8999999999999999很奇怪,并不是我们想要的值0.9 再运行如下代码:System.o ...
随机推荐
- vue运行时 eslint 报“import/first” WARN deprecated browserslist 问题解决
vue运行时 eslint 报“import/first” WARN deprecated browserslist 问题解决 这个信息的意思是导入文件顺序不对,绝对导入应该放在相对导入前面.将绝对 ...
- java static语句的总结
static 是静态方法,他的引用不需要对象,可以使用类名直接进行引用,当然也不需要this. 由于不需要对象,所以static方法内无法调用非static的方法或对象 至于为什么mai ...
- Docker Swarm java 服务集群
Docker Swarm java 服务集群 环境1: 系统:Linux Centos 7.4 x64 内核:Linux docker 3.10.0-693.2.2.el7.x86_64 Docker ...
- Python IO内核交互了解
注:Unix \ Linux 环境下的network IO 用户空间与内核空间 现在操作系统都是采用虚拟存储器,那么对32位操作系统而言,它的寻址空间(虚拟存储空间)为4G(2的32次方).操作系 ...
- Oracle 11gR2 用户重命名(rename user)
Oracle 11.2.0.2里新增了一个新特性——用户重命名(Rename User),在这个版本之前要想重命名用户,需要按用户导出,再fromuser touser(imp)或remap_sc ...
- opencv学习之路(27)、轮廓查找与绘制(六)——外接圆、椭圆拟合、逼近多边形曲线、计算轮廓面积及长度、提取不规则轮廓
一.最小外接圆 #include "opencv2/opencv.hpp" #include<iostream> using namespace std; using ...
- kubernetes endpoint一会消失一会出现的问题剖析
问题现象 发现某个service的后端endpoint一会显示有后端,一会显示没有.显示没有后端,意味着后端的address被判定为notready. endpoint不正常的时候: [root@lo ...
- mysql 5.7 运维命令
MySQL安装 1)卸载旧版本 a) 停掉mysql #pkill -9 mysqld b) 查看rpm包 # rpm -qa|grep -i mysql c) 删除 #yum -y remove m ...
- iOS开发 -------- AFNetworking使用中遇到的小问题
1 在使用的时候会经常出现如下图打印信息,导致请求的数据不能正常显示出来 2 此时找到AFJSONResponseSerializer类,找到箭头标识的位置 3 在下图标识的位置,往集合里面添加 元素 ...
- 单元测试系列之七:Sonar 数据库表关系整理一(rule相关)
更多原创测试技术文章同步更新到微信公众号 :三国测,敬请扫码关注个人的微信号,感谢! 原文链接:http://www.cnblogs.com/zishi/p/7510072.html 简介:Sonar ...