java 将小数拆分为两部分+浮点型精度丢失问题
问题:将一个String类型的小数拆分为整数部分和小数部分,如9.9拆分为9和0.9
1.将小数的整数和小数部分拆分开
public float numberSub(String totalMoney){
float moneyFloat=Float.parseFloat(totalMoney);
System.out.println("moneyFloat="+moneyFloat);
int moneyInteger=(int) moneyFloat;
System.out.println("moneyInteger="+moneyInteger);
float moneyDecimal=moneyFloat-moneyInteger;
System.out.println("moneyDecimal="+moneyDecimal);
float result=this.sub(moneyFloat, moneyInteger);
return result;
}
上面这个方法里面,float-->int转化时直接丢弃小数部分,从而取得小数中的整数,而后作差得到小数部分,但是看下面输出:
2.浮点型表示一个小数的时候存在精度不准确的问题
原因:
首先我们要搞清楚下面两个问题:
() 十进制整数如何转化为二进制数
算法很简单。举个例子,11表示成二进制数:
/=5 余 /= 余 /= 余 /= 余 0结束 11二进制表示为(从下往上): 这里提一点:只要遇到除以后的结果为0了就结束了,大家想一想,所有的整数除以2是不是一定能够最终得到0。换句话说,所有的整数转变为二进制数的算法会不会无限循环下去呢?绝对不会,整数永远可以用二进制精确表示 ,但小数就不一定了。 () 十进制小数如何转化为二进制数
算法是乘以2直到没有了小数为止。举个例子,.9表示成二进制数
0.9*=1.8 取整数部分 0.8(.8的小数部分)*=1.6 取整数部分 0.6*=1.2 取整数部分 0.2*=0.4 取整数部分 0.4*=0.8 取整数部分 0.8*=1.6 取整数部分 0.6*=1.2 取整数部分 ......... .9二进制表示为(从上往下): ...... 注意:上面的计算过程循环了,也就是说*2永远不可能消灭小数部分,这样算法将无限下去。很显然,小数的二进制表示有时是不可能精确的 。其实道理很简单,十进制系统中能不能准确表示出1/3呢?同样二进制系统也无法准确表示1/。这也就解释了为什么浮点型减法出现了精度丢失的问题。
3.验证
众所周知、 Java 的float型在内存中占4个字节。float的32个二进制位结构如下
float内存存储结构
4bytes 31 30 29----23 22----0
表示 实数符号位 指数符号位 指数位 有效数位
其中符号位1表示正,0表示负。有效位数位24位,其中一位是实数符号位。
将一个float型转化为内存存储格式的步骤为:
(1)先将这个实数的绝对值化为二进制格式,注意实数的整数部分和小数部分的二进制方法在上面已经探讨过了。
(2)将这个二进制格式实数的小数点左移或右移n位,直到小数点移动到第一个有效数字的右边。
(3)从小数点右边第一位开始数出二十三位数字放入第22到第0位。
(4)如果实数是正的,则在第31位放入“0”,否则放入“1”。
(5)如果n 是左移得到的,说明指数是正的,第30位放入“1”。如果n是右移得到的或n=0,则第30位放入“0”。
(6)如果n是左移得到的,则将n减去1后化为二进制,并在左边加“0”补足七位,放入第29到第23位。如果n是右移得到的或n=0,则将n化为二进制后在左边加“0”补足七位,再各位求反,再放入第29到第23位。
我们以数字8举例验证,float类型为8.0
1.将8.0转换为二进制之后是1000.0
2.将小数点左移三位到第一个有效位右侧1.0000(保证有效位数24位)得1.00000000000000000000000(只保留24位,多出的被截取掉了,从而引起了误差!)
3.这时已经有了二十四位有效数字,将最左边一位“1”去掉,得到0000000000000000000000共23位,将它放入float存储结构的第22到第0位。
4.因为8.0是正数,因此在第31位实数符号位放入“0”。
5.由于我们把小数点左移,因此在第30位指数符号位放入“1”。
6.因为我们是把小数点左移3位,因此将3减去1得2,化为二进制并补足7位得到0000010,放入第29到第23位。
最后得到0 1 0000010 0000000000000000000000
代码中打印出来如下:
public void testJavaDataType(){
int aint=;
float afloat=;
int fl=Float.floatToIntBits(afloat);
double adouble=;
long dl=Double.doubleToLongBits(adouble);
System.out.println("Int 8 :"+Integer.toBinaryString(aint));
System.out.println("Float 8 :"+Integer.toBinaryString(fl));
System.out.println("Double 8 :"+Long.toBinaryString(dl));
}
打印的时候如果是整数则第31位0默认不打印,篮框是第30位指数符号位,红框是指数位,蓝底是有效位数
很显然对于小数来说一个有限的23位有效位数是不足以精确表示一个小数的。
4.BigDecimal解决
public float sub(float a,float b){
BigDecimal bda=new BigDecimal(Float.toString(a));
BigDecimal bdb=new BigDecimal(String.valueOf(b));
return bda.subtract(bdb).floatValue();
}
5.利用String截取小数点来解决该问题
public float stringSub(String totalMoeny){
String[] strArray=totalMoeny.split("\\.");
if(strArray.length>=){
float a=Float.valueOf(strArray[]);
System.out.println(a);
float b=Float.parseFloat("0."+strArray[1]);
System.out.println(b);
return a-b;
}else{
return 0l;
}
}
参考文章:
http://blog.csdn.net/chencheng19912012/article/details/30072389
http://www.cnblogs.com/jym-sunshine/p/5853070.html
http://blog.csdn.net/tomcat_2014/article/details/51453988
java 将小数拆分为两部分+浮点型精度丢失问题的更多相关文章
- java 取小数点后两位 不四舍五入,怎么做
java 取小数点后两位 不四舍五入,怎么做 正常版: //正常版: import java.text.DecimalFormat; import java.math.RoundingMode; De ...
- Java中String转换Double类型 Java小数点后留两位
Java中String转换Double类型 double num1 = 0.0; String qq = "19.987"; num1 = Double.valueOf(qq.to ...
- 如何把Java的double类型变量保留两位小数
已知 双精度标量 f, 如果想以字符串形式输出,小数点后保留2位,可直接通过C语言的输出格式,System.out.printf("%.2f", f), 达到目的. 如果想要先转 ...
- java取小数点后两位
package com.yonyou.sud.algorithm; import java.math.BigDecimal;import java.text.DecimalFormat;/*** ja ...
- java 保留小数点后N位数(若干位),几种实现的方式总结
import java.math.BigDecimal;import java.text.DecimalFormat;import java.text.NumberFormat;/** * java ...
- Java中产生随机数的两个方法
Java中产生随机数的两个方法 一.利用random方法来生成Java随机数. 在Java语言中生成Java随机数相对来说比较简单,因为有一个现成的方法可以使用.在Math类中,Java语言提供了一个 ...
- Java页面中文编码要转换两次encodeURI
1.js文件中使用encodeURI()方法. login_name = encodeURI(encodeURI(login_name)); 2.action中URLDecoder解码 loginNa ...
- 分列:将excel单元格的内容拆分为两列
提要:处理excel数据时有时需要把单元格的内容拆分为两列,可能方便外部软件的链接,可能使数据显示更明晰等等,有人说直接剪切加粘贴不就可以了吗,但是有时数据过多,这样处理很不效率,网上搜索的方法说插入 ...
- Java中HashMap遍历的两种方式
Java中HashMap遍历的两种方式 转]Java中HashMap遍历的两种方式原文地址: http://www.javaweb.cc/language/java/032291.shtml 第一种: ...
随机推荐
- Sublime Text 模版插件: SublimeTmpl
开发者的插件介绍页面:http://www.fantxi.com/blog/archives/sublime-template-engine-sublimetmpl/ 写了个sublime的模版插件, ...
- YII2与Thinkphp整合阿里云OSS
前言: 如果上传的文件都和网站程序源代码放在一起:那是有相当多的弊端的: 1:静态文件会占用大量带宽: 2:服务器的成本略高: 常规的做法是把php源代码放到一台服务器上:图片等静态文件放在另一台服务 ...
- PHP实现手机号码中间四位用星号(*)隐藏的自定义函数分享
php屏蔽电话号码中间四位: Method 1: function hidtel($phone){ $IsWhat = preg_match('/(0[0-9]{2,3}[\-]?[2-9][0-9] ...
- hdu 1058:Humble Numbers(动态规划 DP)
Humble Numbers Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)To ...
- MVC模式 与 Model2模型 介绍
Model1回顾 MVC模式:MVC(Model.View.Controller)是软件开发过程中比较流行的设计思想.旨在分离模型.控制.师徒.是一种分层思想的体现. Model2简介Java Web ...
- excel如何将一个工作薄中的工作表生成独立的工作薄
excel如何将一个工作薄中的工作表生成独立的工作薄 '用vba代码 Sub 另存所有工作表为工作簿() Dim sht As Worksheet Application.ScreenUpdatin ...
- 【黑金原创教程】【TimeQuest】【第七章】供源时钟与其他
声明:本文为黑金动力社区(http://www.heijin.org)原创教程,如需转载请注明出处,谢谢! 黑金动力社区2013年原创教程连载计划: http://www.cnblogs.com/al ...
- GitHubDesktop的使用方法
author:headsen chen date:2018-05-30 17:24:55 notice:This article is created by headsen chen hims ...
- Animate a custom Dialog,自定义Dialog动画
Inside res/style.xml <style name="AppTheme" parent="android:Theme.Light" /> ...
- CSS3制作旋转导航
慕课网学习CSS3时,遇到个习题,觉得有必要总结学习下:CSS3制作旋转导航 慕课网习题地址:http://www.imooc.com/code/1883 示例及源码地址:http://codepen ...