黑马程序员技术交流社区

标题: 关于double类型的问题 [打印本页]

作者: lxww    时间: 2013-3-22 22:52
标题: 关于double类型的问题
public class Test11 {
        public static void main(String[] args) {
                double a=1.4;
                double b=1.5;
                System.out.println(a*b);
        }
}
输出的结果:2.0999999999999996

请问:这是为什么呢?很多时候牵扯到double类型的时候,总是会有很多问题。今天看到基础题目,但是运行的实际结果让人难以捉摸。double类型的数据怎么这么纠结。
求解释。顺便详细说说double类型。要有深度。
作者: 我自信我很牛    时间: 2013-3-22 23:01
这个问题太给力了,坐等大神回复。。。
作者: 李晓峰    时间: 2013-3-22 23:15
{:soso_e179:}坚决顶起来,期待高手解答。
作者: 范鹏飞    时间: 2013-3-22 23:28
java里面double和float类型是非精确数字类型,存在一定的误差,误差值具有随机性,如果要精确计算的,尽量避免用double和float
作者: 范鹏飞    时间: 2013-3-22 23:29
拷贝的
浮点运算很少是精确的,只要是超过精度能表示的范围就会产生误差。往往产生误差不是因为数的大小,而是因为数的精度。因此,产生的结果接近但不等于想要的结果。尤其在使用 float 和 double 作精确运算的时候要特别小心。
可以考虑采用一些替代方案来实现。如通过 String 结合 BigDecimal 或者通过使用 long 类型来转换。
作者: 葛伟    时间: 2013-3-22 23:30
这个我也是在百度搜到的

使用Java,double 进行运算时,经常出现精度丢失的问题,总是在一个正确的结果左右偏0.0000**1。 特别在实际项目中,通过一个公式校验该值是否大于0,如果大于0我们会做一件事情,小于0我们又处理其他事情。 这样的情况通过double计算出来的结果去和0比较大小,尤其是有小数点的时候,经常会因为精度丢失而导致程序处理流程出错。
     所以一般对double类型进行运算时,做好对结果进行处理,然后拿这个值去做其他事情。

     目前总结如下:

     /**  
     * 对double数据进行取精度.  
     * @param value  double数据.  
     * @param scale  精度位数(保留的小数位数).  
     * @param roundingMode  精度取值方式.  
     * @return 精度计算后的数据.  
     */  
    public static double round(double value, int scale,
             int roundingMode) {   
        BigDecimal bd = new BigDecimal(value);   
        bd = bd.setScale(scale, roundingMode);   
        double d = bd.doubleValue();   
        bd = null;   
        return d;   
    }   


     /**
     * double 相加
     * @param d1
     * @param d2
     * @return
     */
    public double sum(double d1,double d2){
        BigDecimal bd1 = new BigDecimal(Double.toString(d1));
        BigDecimal bd2 = new BigDecimal(Double.toString(d2));
        return bd1.add(bd2).doubleValue();
    }


    /**
     * double 相减
     * @param d1
     * @param d2
     * @return
     */
    public double sub(double d1,double d2){
        BigDecimal bd1 = new BigDecimal(Double.toString(d1));
        BigDecimal bd2 = new BigDecimal(Double.toString(d2));
        return bd1.subtract(bd2).doubleValue();
    }

    /**
     * double 乘法
     * @param d1
     * @param d2
     * @return
     */
    public double mul(double d1,double d2){
        BigDecimal bd1 = new BigDecimal(Double.toString(d1));
        BigDecimal bd2 = new BigDecimal(Double.toString(d2));
        return bd1.multiply(bd2).doubleValue();
    }


    /**
     * double 除法
     * @param d1
     * @param d2
     * @param scale 四舍五入 小数点位数
     * @return
     */
    public double div(double d1,double d2,int scale){
        //  当然在此之前,你要判断分母是否为0,   
        //  为0你可以根据实际需求做相应的处理

        BigDecimal bd1 = new BigDecimal(Double.toString(d1));
        BigDecimal bd2 = new BigDecimal(Double.toString(d2));
        return bd1.divide
               (bd2,scale,BigDecimal.ROUND_HALF_UP).doubleValue();
    }


这样,计算double类型的数据计算问题就可以处理了。
另外补充一下 JavaScript 四舍五入的方法:
小数点问题
Math.round(totalAmount*100)/100 (保留 2 位)

function formatFloat(src, pos)
{
  return Math.round(src*Math.pow(10, pos))/Math.pow(10, pos);
}


作者: 黑马19我最牛    时间: 2013-3-22 23:32
http://wenku.baidu.com/view/9f00536c25c52cc58bd6be86.html

不好说,先看下这个网页上面讲的吧.我不太能看懂,不知道你们能不能看懂,看懂的,希望在这里说下,当然有大神更好!




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2