[本文相关的代码放在github上。地址为:https://github.com/VigourJiang/StructuredFloat]

Java中double类型的格式基本遵循IEEE 754标准。

虽然数学意义上的小数是连续的。但double只能表示当中的一些离散点。把这些离散点组成的集合记为S。S的大小还是有限的。假设要保存的小数P刚好在集合S内,那么double类型就能精确的表示P;否则double类型只能从集合S中找一个与P近期的离散点P'取代P。

以上表述对于float也成立。IEEE 754中float和double的表示策略全然同样。差别只体如今各个字段(指数字段、小数字段)的bit数量不同。

也就是说。float和double都是不精确的,因此偶尔会有一些奇怪的事情发生:

       double d = 1e16d;

       double d2 = d +1.0d - d;

       // 下面代码输出0.0,而不是1.0
System.out.println(d2);

上述结果或许并不有趣。由于在学C语言的时候。你的老师就可能提醒过你。但有趣的是,float和double到底有多不精确?给定一个double d = XXX,与d最接近的另外两个double的数值是多少?

我用Java写了一个解析原始数据类型float和double的类StructuredFloat,给定一个float或者double数值P。它能够计算出与P的绝对值近期的、能够被float/double表示的浮点数。以下是使用方法:

	    double cur = 3.23d; // any valid double value
StructuredFloat sd = com.vigour.StructuredFloat.StructedFloatBuilder.buildDouble(cur);
StructuredFloat smaller = sd.absSmaller();
StructuredFloat bigger = sd.absBigger(); if(smaller != null){
double smaller_double = smaller.getBigDecimal().doubleValue();
// now you get the nearest double value whose absolute value is smaller
}
if(bigger!= null){
double bigger_double = bigger.getBigDecimal().doubleValue();
// now you get the nearest double value whose absolute value is bigger
}

以下是一些有趣的输出,能够看到,在1e16d附近double的精度就小于1了,在1e7f(一千万)附近。float的精度就等于1了,float果然难堪大用。

-------------Some Interesting Resule for Double--------------------
// 0.1d附近的double
Nearest smaller double: 0.09999999999999999167332731531132594682276248931884765625
Current double: 0.1000000000000000055511151231257827021181583404541015625
Nearest bigger double : 0.10000000000000001942890293094023945741355419158935546875 // 1.0d附近的double
Nearest smaller double: 0.99999999999999988897769753748434595763683319091796875
Current double: 1
Nearest bigger double : 1.0000000000000002220446049250313080847263336181640625 // 10.0d附近的double
Nearest smaller double: 9.9999999999999982236431605997495353221893310546875000
Current double: 10.00
Nearest bigger double : 10.0000000000000017763568394002504646778106689453125000 // 1e14d附近的double
Nearest smaller double: 99999999999999.9843750000000000000000000000000000000000000000000000
Current double: 100000000000000.00000000000000000000000000000000
Nearest bigger double : 100000000000000.0156250000000000000000000000000000000000000000000000 // 1e15d附近的double
Nearest smaller double: 999999999999999.8750000000000000000000000000000000000000000000000000
Current double: 1000000000000000.0000000000000000000000000000000000
Nearest bigger double : 1000000000000000.1250000000000000000000000000000000000000000000000000 // 1e16d附近的double
Nearest smaller double: 9999999999999998.0000000000000000000000000000000000000000000000000000
Current double: 10000000000000000.0000000000000000000000000000000000000
Nearest bigger double : 10000000000000002.0000000000000000000000000000000000000000000000000000 // 1e17d附近的double
Nearest smaller double: 99999999999999984.0000000000000000000000000000000000000000000000000000
Current double: 100000000000000000.000000000000000000000000000000000000000
Nearest bigger double : 100000000000000016.0000000000000000000000000000000000000000000000000000 // 1e304d附近的double
Nearest smaller double: 9999999999999998174371273630364736815867488735718786093662414371947263704524926751224722911637244940234972882804879769415602664816552507597839565690480126952738889402600333599657997758603312171995012866291845554976690497648524473448849371595248581587050582985041870802940253992811266476846330599148879872.0000000000000000000000000000000000000000000000000000
Current double: 9999999999999999392535525055364621860040287220117324953190771571323204563013233902843309257440507748436856118056162172578717193742636030530235798840866882774987301441682011041067710253162440905843719802548551599076639682550821832659549112269607949805346034918662572406407604380845959862074904348138143744.000000000000000000000000000000000000000000000000
Nearest bigger double : 10000000000000000610699776480364506904213085704515863812719128770699145421501541054461895603243770556638739353307444575741831722668719553462632031991253638597235713480763688482477422747721569639692426738805257643176588867453119191870248852943967318023641486852283274009874954768880653247303478097127407616.0000000000000000000000000000000000000000000000000000 -------------Some Interesting Resule for Float-------------------- // 0.1f附近的float
Nearest smaller float: 0.0999999940395355224609375
Current float: 0.100000001490116119384765625
Nearest bigger float: 0.10000000894069671630859375 // 1.0f附近的float
Nearest smaller float: 0.999999940395355224609375
Current float: 1
Nearest bigger float: 1.00000011920928955078125 // 10.0f附近的float
Nearest smaller float: 9.99999904632568359375000
Current float: 10.00
Nearest bigger float: 10.00000095367431640625000 // 1e5f附近的float
Nearest smaller float: 99999.99218750000000000000000
Current float: 100000.00000000000
Nearest bigger float: 100000.00781250000000000000000 // 1e6f附近的float
Nearest smaller float: 999999.93750000000000000000000
Current float: 1000000.0000000000000
Nearest bigger float: 1000000.06250000000000000000000 // 1e7f附近的float
Nearest smaller float: 9999999.00000000000000000000000
Current float: 10000000.0000000000000000
Nearest bigger float: 10000001.00000000000000000000000 // 1e8f附近的float
Nearest smaller float: 99999992.00000000000000000000000
Current float: 100000000.000000000000000000
Nearest bigger float: 100000008.00000000000000000000000 // 1e38f附近的float
Nearest smaller float: 99999986661652122824821048795547566080.00000000000000000000
Current float: 99999996802856924650656260769173209088.00000000000000000000000
Nearest bigger float: 100000006944061726476491472742798852096.0000000000000000000000

精确计算java中float和double的精度的更多相关文章

  1. Java中 float、double使用注意问题

    在java中运行一下代码 System.out.println(2.00-1.10);输出的结果是:0.8999999999999999很奇怪,并不是我们想要的值0.9 再运行如下代码:System. ...

  2. Java中float和double转换的问题

    为什么double转float不会出现数据误差,而float转double却误差如此之大?   double d = 3.14; float f = (float)d; System.out.prin ...

  3. Java中float、double、long类型变量赋值添加f、d、L尾缀问题

    展开1. 添加尾缀说明 我们知道Java在变量赋值的时候,其中float.double.long数据类型变量,需要在赋值直接量后面分别添加f或F.d或D.l或L尾缀来说明.  其中,long类型最好以 ...

  4. JAVA中float与double的区别

    float是单精度类型,精度是8位有效数字,取值范围是10的-38次方到10的38次方,float占用4个字节的存储空间 double是双精度类型,精度是17位有效数字,取值范围是10的-308次方到 ...

  5. java中float和double的区别

    float表示单精度浮点数在机内占4个字节,用32位二进制描述. double表示双精度浮点数在机内占8个字节,用64位二进制描述.浮点数在机内用指数型式表示,分解为:数符,尾数,指数符,指数四部分. ...

  6. 【转】JAVA程序中Float和Double精度丢失问题

    原文网址:http://blog.sina.com.cn/s/blog_827d041701017ctm.html 问题提出:12.0f-11.9f=0.10000038,"减不尽" ...

  7. Java中float/double取值范围与精度

    Java浮点数 浮点数结构 要说清楚Java浮点数的取值范围与其精度,必须先了解浮点数的表示方法,浮点数的结构组成,之所以会有这种所谓的结构,是因为机器只认识01,你想表示小数,你要机器认识小数点这个 ...

  8. 【转载】 C#中float、double以及decimal类型有何不同

    在C#语言中,float.double以及decimal类型都可以用来表示小数,但三者还是有一定的不同,有效数字为相比的话,decimal类型的有效数字最大,float类型最小.计算浮点类型的运算,如 ...

  9. Java中long和double的原子性

    Java中long和double的原子性 java中基本类型中,long和double的长度都是8个字节,32位(4字节)处理器对其读写操作无法一次完成,那么,JVM,long和double是原子性的 ...

随机推荐

  1. Android屏幕尺寸与度量单位(px,dp,sp)简介

    MarkdownPad Document *:first-child { margin-top: 0 !important; } body>*:last-child { margin-botto ...

  2. 【sqli-labs】 less64 GET -Challenge -Blind -130 queries allowed -Variation3 (GET型 挑战 盲注 只允许130次查询 变化3)

    双括号整型 http://192.168.136.128/sqli-labs-master/Less-64/?id=1)) or ((1

  3. 动软生成器添加Mysql注释

    1.解决没有mysql注释问题 替换原文件下载地址 2.更新Models模板 <#@ template language="c#" HostSpecific="Tr ...

  4. 扩增子统计绘图1箱线图:Alpha多样性

    绘制Alpha多样性线箱图 绘图和统计全部为R语言,建议复制代码,在Rstuido中运行,并设置工作目录为存储之前分析结果文件的result目录 # 运行前,请在Rstudio中菜单栏选择“Sessi ...

  5. Redis系列(十)--集群cluster

    在之前学习了Master-Slave.Sentinel模式,但是在某些情况下还是无法满足系统对QPS等要求,这时候就需要Cluster,Redis3.0支持了cluster 一.为什么使用Cluste ...

  6. vue组件---动态组件之多标签页面

    首先看下效果图 代码: <!DOCTYPE html> <html> <head> <meta charset="utf-8"> & ...

  7. oracle 安装准备

    1.选择数据库 (官网查询支持的操作系统) 2.选择系统 (官网查询支持的硬件)(更新补丁) 3.选择硬件 (io性能测试--oracle 大量小文件读写) 4.oracle 升级(和打补丁) 5.o ...

  8. Uploadify上传大文件

    一丶参考地址 <script type="text/javascript"> var auth = "@(Request.Cookies[FormsAuthe ...

  9. Java排序算法全

    目录 Java排序算法代码 零. 排序基类 一. 选择排序 二. 插入排序 三. 希尔排序 四. 归并排序 1. 自顶向下 2. 自底向上 五. 快速排序 1. 基本版 2. 双路切分版 3. 三路切 ...

  10. 洛谷——P3173 [HAOI2009]巧克力

    P3173 [HAOI2009]巧克力 题目描述 有一块n*m的矩形巧克力,准备将它切成n*m块.巧克力上共有n-1条横线和m-1条竖线,你每次可以沿着其中的一条横线或竖线将巧克力切开,无论切割的长短 ...