Java浮点数详解(double float)-- 分分搞定浮点数

初学Java只简单听说double,float是不能精确的表示一个数,然后留下满满的疑问? why? 这个问题已经放了好久时间了,今天终于把这个问题整明白了!首先整理一下以下几个困惑


1.浮点数(double  float)真的会不精确?

我们在使用的过程中并没有误差啊?

double a = 0.4;

double b = a*4;

System.out.println(b); // 输出 1.6

// 不是说浮点数不精确吗?那我们做一个相等判断

System.out.println(a == 0.4); //输出 true

// 我去,哪有什么不精确!!!瞎BB

当我们放心大胆的用浮点数时

System.out.println (0.4 * 0.2);

// 输出 0.08000000000000002

System.out.println(40000000.0f == 40000000.5f); // float类型字面量

// 输出 true

System.out.println(4000000000000000.0 == 4000000000000000.2);

// 输出 true

// 啊西巴! 什么情况

2.浮点数为什么会不精确?

粗浅的解释:

1. double 的取值范围 [-1.79 e+308 , 1.79 e+308] 。但 double 的存储空间为 8 字节一共 64 位,也就是说 double 最多可以用 2∧64 个不同值来表示不同区间,约等于 1.84 e+19 个区间。

这也是为什么 40000000.0 f = 40000000.5 f, 4000000000000000.0 = 4000000000000000.2 ,因为他们在同一个区间。

2. 每一个区间都会用一个值来显示,但这个显示值(在这个区间的值都用这个值显示)与存储值(存储在计算机中二进制所表示的值)并不一定相等。(“ 整数 ” 显示值 = 储存值,所以下面只讨论 “ 小数 ”)


显示值  (double)   储存值 (double)

0.5     0.5

0.4     0.40000000000000002220446049250313080847263336181640625

0.25    0.25

0.2     0.200000000000000011102230246251565404236316680908203125


什么时候 显示值 会与 储存值 相等呢? 这里有一个规律

有一个小于 1 的小数 d ,如果满足 d*2∧n = 1 则 d 的 显示值 等与 储存值 。( n 为正整数) 

比如:                0.5 * 2 = 1 ,               0.25 * 2 * 2  = 1 ,             0.125 * 2 * 2 * 2 = 1

为什么会有这个规律,可以了解小数的二进制表示 。

计算和比较时用的是 储存值,最终以 计算结果所在区间的 显示值 显示在屏幕上;


显示值  (double)    储存值 (double)

0.4      0.40000000000000002220446049250313080847263336181640625

0.2      0.200000000000000011102230246251565404236316680908203125


接下来我们可以解释:0.4   *   0.2   =   0.08000000000000002 

a  = (0.4的储存值) 0.40000000000000002220446049250313080847263336181640625;

b  = (0.2的储存值) 0.200000000000000011102230246251565404236316680908203125;

a * b ≈ 0.08000000000000000888178419700125256990808622629275169;

计算结果需要存储,它所在区间对应的 储存值 为 :

0.080000000000000015543122344752191565930843353271484375

该 存储值 对应的 显示值 为:

0.08000000000000002

3.误差累积效应(重点)

浮点数因为计算时是用 储存值 多次计算,有可能会放大误差。

System.out.println(0.4 * 0.4 * 0.4); // 输出 0.06400000000000002

System.out.println(0.5 * 0.5 * 0.5); // 输出 0.125

注意是有可能不是一定,有两种情况;

1. (显示值  =  储存值) ,这样多次计算没有误差,除非计算中出现数值的有效位数过长,double类型不能表示,会出现精度丢失的情况

2. (显示值  >  储存值) 与 (显示值  <  储存值)混合运算,它们的误差会相互抵消

3.浮点数使用场景

一般来说浮点数使用比较方便,误差比较小( 可以忽略 ),计算机操作浮点数的效率高( 与BigDecimal相比较),所以浮点数挺常用的。但使用它必须 慎重 考虑以下情况:

1. 数值的有效位数过长,会发生精度丢失;

2. 数值较大,误差也会变大了;

    0.40000000000000000000000    =    0.40000000000000002220446

(在同一个区间,储存值都为  0.40000000000000002220446049250313080847263336181640625  所以相等)

    40000000000000000000000.0     =   40000000000000002220446.0

(在同一个区间,储存值都为  40000000000000000000000  所以相等)

3. 浮点数的 误差累积效应;

4. 该数值表示的类型是否对误差有严格的要求,(比如:时间,金钱)

4.该如何进行精确的计算

Java要进行精确计算,需要使用到类java.math.Big。借助BigDecima可以查看浮点数的储存值;

BigDecimal b1 = new BigDecimal( 0.4 ); // double类型字面量

BigDecimal b2 = new BigDecimal( 0.4f ); // float类型字面量

System.out.println( b1 ); // 输出 0.40000000000000002220446049250313080847263336181640625

System.out.println( b2 ); // 输出 0.4000000059604644775390625

哇卡卡,通过上面的案例,可以发现 BigDecima 真强大!!不过用它做计算有点麻烦

publicBigDecimal add(BigDecimal value);//加法

publicBigDecimal subtract(BigDecimal value);//减法

publicBigDecimal multiply(BigDecimal value);//乘法

publicBigDecimal divide(BigDecimal value);//除法

BigDecimal b1 = new BigDecimal("0.4");

BigDecimal b2 = new BigDecimal("0.2");

System.out.println(b1.add(b2)); // 输出 0.6

System.out.println(b1.subtract(b2)); // 输出 0.2

System.out.println(b1.multiply(b2)); // 输出 0.08

System.out.println(b1.divide(b2)); // 输出 2

若要详细了解BigDecima,请查看API。



最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容