infofee=infofee/100 ;
        sxfee=infofee*Integer.parseInt(sxlv) *(0.01);
        feeload=Integer.parseInt(feelv);
        txfee=(nummt-nummo)*feeload*0.01;
        fee=(infofee-sxfee-txfee)*Integer.parseInt(fclv) *0.01;
        aainfofee=aainfofee+infofee;
        aanummt=aanummt+nummt;
        aanummo=aanummo+nummo;
        aasxfee=aasxfee+sxfee;
        aatxfee=aatxfee+txfee;
        aafee=aafee+fee;
        f1=Double.toString(infofee);
        f1=f1.substring(0,f1.indexOf('.')+2);
        f2=Double.toString(sxfee);
        f2=f2.substring(0,f2.indexOf('.')+2);
        f3=Double.toString(txfee);
        f3=f3.substring(0,f3.indexOf('.')+2);
        f4=Double.toString(fee);
        f4=f4.substring(0,f4.indexOf('.')+2);以前做的程序的一段

解决方案 »

  1.   

    Java 理论与实践:您的小数点到哪里去了? 
    使用浮点数和小数中的技巧和陷阱
    级别:中级
    Brian Goetz([email protected]
    首席顾问,Quiotix Corp
    2003 年 4 月许多程序员在其整个开发生涯中都不曾使用定点或浮点数,可能的例外是,偶尔在计时测试或基准测试程序中会用到。Java 语言和类库支持两类非整数类型 — IEEE 754 浮点(float 和 double,包装类(wrapper class)为 Float 和 Double),以及任意精度的小数(java.math.BigDecimal)。在本月的 Java 理论和实践中,Brian Goetz 探讨了在 Java 程序中使用非整数类型时一些常碰到的陷阱和“gotcha”。请在本文的论坛上提出您对本文的想法,以飨笔者和其他读者。(您也可以单击本文顶部或底部的讨论来访问论坛 )。
    虽然几乎每种处理器和编程语言都支持浮点运算,但大多数程序员很少注意它。这容易理解 — 我们中大多数很少需要使用非整数类型。除了科学计算和偶尔的计时测试或基准测试程序,其它情况下几乎都用不着它。同样,大多数开发人员也容易忽略 java.math.BigDecimal 所提供的任意精度的小数 — 大多数应用程序不使用它们。然而,在以整数为主的程序中有时确实会出人意料地需要表示非整型数据。例如,JDBC 使用 BigDecimal 作为 SQL DECIMAL 列的首选互换格式。IEEE 浮点
    Java 语言支持两种基本的浮点类型:float 和 double,以及与它们对应的包装类 Float 和 Double。它们都依据 IEEE 754 标准,该标准为 32 位浮点和 64 位双精度浮点二进制小数定义了二进制标准。IEEE 754 用科学记数法以底数为 2 的小数来表示浮点数。IEEE 浮点数用 1 位表示数字的符号,用 8 位来表示指数,用 23 位来表示尾数,即小数部分。作为有符号整数的指数可以有正负之分。小数部分用二进制(底数 2)小数来表示,这意味着最高位对应着值 ?(2-1),第二位对应着 ?(2-2),依此类推。对于双精度浮点数,用 11 位表示指数,52 位表示尾数。IEEE 浮点值的格式如图 1 所示。图 1. IEEE 754 浮点数的格式
    因为用科学记数法可以有多种方式来表示给定数字,所以要规范化浮点数,以便用底数为 2 并且小数点左边为 1 的小数来表示,按照需要调节指数就可以得到所需的数字。所以,例如,数 1.25 可以表示为尾数为 1.01,指数为 0:
    (-1)0*1.012*20数 10.0 可以表示为尾数为 1.01,指数为 3:
    (-1)0*1.012*23特殊数字
    除了编码所允许的值的标准范围(对于 float,从 1.4e-45 到 3.4028235e+38),还有一些表示无穷大、负无穷大、-0 和 NaN(它代表“不是一个数字”)的特殊值。这些值的存在是为了在出现错误条件(譬如算术溢出,给负数开平方根,除以 0 等)下,可以用浮点值集合中的数字来表示所产生的结果。这些特殊的数字有一些不寻常的特征。例如,0 和 -0 是不同值,但在比较它们是否相等时,被认为是相等的。用一个非零数去除以无穷大的数,结果等于 0。特殊数字 NaN 是无序的;使用 ==、< 和 > 运算符将 NaN 与其它浮点值比较时,结果为 false。如果 f 为 NaN,则即使 (f == f) 也会得到 false。如果想将浮点值与 NaN 进行比较,则使用 Float.isNaN() 方法。表 1 显示了无穷大和 NaN 的一些属性。表 1. 特殊浮点值的属性表达式 结果 
    Math.sqrt(-1.0) -> NaN 
    0.0 / 0.0 -> NaN 
    1.0 / 0.0 -> 无穷大 
    -1.0 / 0.0 -> 负无穷大 
    NaN + 1.0 -> NaN 
    无穷大 + 1.0 -> 无穷大 
    无穷大 + 无穷大 -> 无穷大 
    NaN > 1.0 -> false 
    NaN == 1.0 -> false 
    NaN < 1.0 -> false 
    NaN == NaN -> false 
    0.0 == -0.01 -> true 基本浮点类型和包装类浮点有不同的比较行为
    使事情更糟的是,在基本 float 类型和包装类 Float 之间,用于比较 NaN 和 -0 的规则是不同的。对于 float 值,比较两个 NaN 值是否相等将会得到 false,而使用 Float.equals() 来比较两个 NaN Float 对象会得到 true。造成这种现象的原因是,如果不这样的话,就不可能将 NaN Float 对象用作 HashMap 中的键。类似的,虽然 0 和 -0 在表示为浮点值时,被认为是相等的,但使用 Float.compareTo() 来比较作为 Float 对象的 0 和 -0 时,会显示 -0 小于 0。浮点中的危险
    由于无穷大、NaN 和 0 的特殊行为,当应用浮点数时,可能看似无害的转换和优化实际上是不正确的。例如,虽然好象 0.0-f 很明显等于 -f,但当 f 为 0 时,这是不正确的。还有其它类似的 gotcha,表 2 显示了其中一些 gotcha。表 2. 无效的浮点假定这个表达式…… 不一定等于…… 当…… 
    0.0 - f -f f 为 0 
    f < g ! (f >= g) f 或 g 为 NaN 
    f == f true f 为 NaN 
    f + g - g f g 为无穷大或 NaN 舍入误差
    浮点运算很少是精确的。虽然一些数字(譬如 0.5)可以精确地表示为二进制(底数 2)小数(因为 0.5 等于 2-1),但其它一些数字(譬如 0.1)就不能精确的表示。因此,浮点运算可能导致舍入误差,产生的结果接近 — 但不等于 — 您可能希望的结果。例如,下面这个简单的计算将得到 2.600000000000001,而不是 2.6:double s=0;
    for (int i=0; i<26; i++)
    s += 0.1;
    System.out.println(s);
    类似的,.1*26 相乘所产生的结果不等于 .1 自身加 26 次所得到的结果。当将浮点数强制转换成整数时,产生的舍入误差甚至更严重,因为强制转换成整数类型会舍弃非整数部分,甚至对于那些“看上去似乎”应该得到整数值的计算, 也存在此类问题。例如,下面这些语句:double d = 29.0 * 0.01;
    System.out.println(d);
    System.out.println((int) (d * 100));
    将得到以下输出:0.29
    28
      

  2.   

    这可能不是您起初所期望的。浮点数比较指南
    由于存在 NaN 的不寻常比较行为和在几乎所有浮点计算中都不可避免地会出现舍入误差,解释浮点值的比较运算符的结果比较麻烦。最好完全避免使用浮点数比较。当然,这并 不总是可能的,但您应该意识到要限制浮点数比较。如果必须比较浮点数来看它们是否相等,则应该将它们差的绝对值同一些预先选定的小正数进行比较,这样您所做的就是测试它们 是否“足够接近”。(如果不知道基本的计算范围,可以使用测试“abs(a/b - 1) < epsilon”,这种方法比简单地比较两者之差要更准确)。甚至测试看一个值是比零大还是比零小也存在危险 —“以为”会生成比零略大值的计算事实上可能由于积累的舍入误差会生成略微比零小的数字。NaN 的无序性质使得在比较浮点数时更容易发生错误。当比较浮点数时,围绕无穷大和 NaN 问题,一种避免 gotcha 的经验法则是显式地测试值的有效性,而不是试图排除无效值。在清单 1 中,有两个可能的用于特性的 setter 的实现,该特性只能接受非负数值。第一个实现会接受 NaN,第二个不会。第二种形式比较好,因为它显式地检测了您认为有效的值的范围。清单 1. 需要非负浮点值的较好办法和较差办法 // Trying to test by exclusion -- this doesn't catch NaN or infinity
    public void setFoo(float foo) {
    if (foo < 0)
    throw new IllegalArgumentException(Float.toString(f));
    this.foo = foo;
    }
    // Testing by inclusion -- this does catch NaN
    public void setFoo(float foo) {
    if (foo >= 0 && foo < Float.INFINITY)
    this.foo = foo;
    else
    throw new IllegalArgumentException(Float.toString(f));
    }
    不要用浮点值表示精确值
    一些非整数值(如几美元和几美分这样的小数)需要很精确。浮点数不是精确值,所以使用它们会导致舍入误差。因此,使用浮 点数来试图表示象货币量这样的精确数量不是一个好的想法。使用浮点数来进行美元和美分计算会得到灾难性的后果。浮点数最好用来表示象测量值这类数值,这类值从一开始就不怎 么精确。用于较小数的 BigDecimal
    从 JDK 1.3 起,Java 开发人员就有了另一种数值表示法来表示非整数:BigDecimal。BigDecimal 是标准的类,在编译器中不需要特殊支持,它可以表示任意精度的小数,并对它们进行计算。在内部,可以用任意精度任何范围的值和一个换算因子来表示 BigDecimal,换算因子表示左移小数点多少位,从而得到所期望范围内的值。因此,用 BigDecimal 表示的数的形式为 unscaledValue*10-scale。用于加、减、乘和除的方法给 BigDecimal 值提供了算术运算。由于 BigDecimal 对象是不可变的,这些方法中的每一个都会产生新的 BigDecimal 对象。因此,因为创建对象的开销,BigDecimal 不适合于大量的数学计算,但设计它的目的是用来精确地表示小数。如果您正在寻找一种能精确表示如货币量这样的数值,则 BigDecimal 可以很好地胜任该任务。所有的 equals 方法都不能真正测试相等
    如浮点类型一样,BigDecimal 也有一些令人奇怪的行为。尤其在使用 equals() 方法来检测数值之间是否相等时要小心。equals() 方法认为,两个表示同一个数但换算值不同(例如,100.00 和 100.000)的 BigDecimal 值是不相等的。然而,compareTo() 方法会认为这两个数是相等的,所以在从数值上比较两个 BigDecimal 值时,应该使用 compareTo() 而不是 equals()。另外还有一些情形,任意精度的小数运算仍不能表示精确结果。例如,1 除以 9 会产生无限循环的小数 .111111...。出于这个原因,在进行除法运算时,BigDecimal 可以让您显式地控制舍入。movePointLeft() 方法支持 10 的幂次方的精确除法。使用 BigDecimal 作为互换类型
    SQL-92 包括 DECIMAL 数据类型,它是用于表示定点小数的精确数字类型,它可以对小数进行基本的算术运算。一些 SQL 语言喜欢称此类型为 NUMERIC 类型,其它一些 SQL 语言则引入了 MONEY 数据类型,MONEY 数据类型被定义为小数点右侧带有两位的小数。如果希望将数字存储到数据库中的 DECIMAL 字段,或从 DECIMAL 字段检索值,则如何确保精确地转换该数字?您可能不希望使用由 JDBC PreparedStatement 和 ResultSet 类所提供的 setFloat() 和 getFloat() 方法,因为浮点数与小数之间的转换可能会丧失精确性。相反,请使用 PreparedStatement 和 ResultSet 的 setBigDecimal() 及 getBigDecimal() 方法。类似的,象 Castor 这样的 XML 数据绑定工具使用 BigDecimal 会生成小数值属性和元素(在 XSD 模式中支持这种基本数据类型)的 getter 和 setter。构造 BigDecimal 数
    对于 BigDecimal,有几个可用的构造函数。其中一个构造函数以双精度浮点数作为输入,另一个以整数和换算因子作为输入,还有一个以小数的 String 表示作为输入。要小心使用 BigDecimal(double) 构造函数,因为如果不了解它,会在计算过程中产生舍入误差。请使用基于整数或 String 的构造函数。如果使用 BigDecimal(double) 构造函数不恰当,在传递给 JDBC setBigDecimal() 方法时,会造成似乎很奇怪的 JDBC 驱动程序中的异常。例如,考虑以下 JDBC 代码,该代码希望将数字 0.01 存储到小数字段:PreparedStatement ps =
    connection.prepareStatement("INSERT INTO Foo SET name=?, value=?");
    ps.setString(1, "penny");
    ps.setBigDecimal(2, new BigDecimal(0.01));
    ps.executeUpdate();
    在执行这段似乎无害的代码时会抛出一些令人迷惑不解的异常(这取决于具体的 JDBC 驱动程序),因为 0.01 的双精度近似值会导致大的换算值,这可能会使 JDBC 驱动程序或数据库感到迷惑。JDBC 驱动程序会产生异常,但可能不会说明代码实际上错在哪里,除非意识到二进制浮点数的局限性。相反,使用 BigDecimal("0.01") 或 BigDecimal(1, 2) 构造 BigDecimal 来避免这类问题,因为这两种方法都可以精确地表示小数。结束语
    在 Java 程序中使用浮点数和小数充满着陷阱。浮点数和小数不象整数一样“循规蹈矩”,不能假定浮点计算一定产生整型或精确的结果,虽然它们的确“应该”那样做。最好将浮点运算保 留用作计算本来就不精确的数值,譬如测量。如果需要表示定点数(譬如,几美元和几美分),则使用 BigDecimal。关于作者
    Brian Goetz 是一位软件顾问,在过去的 15 年里一直是一位专业软件开发人员。他是 Quiotix 的首席顾问,该公司是位于加利福尼亚 Los Altos 的软件开发和咨询公司。请在业界流行的出版物上查阅 Brian 已发表的和即将发表的文章。可以通过 [email protected] 与 Brian 联系。 
      

  3.   

    BigDecimal  试试先
    谢了!!不知道有没有更好的方法
      

  4.   

    to hotenM(南京) :这方法行不通的。 应为“无法精确表示”的数,其失去精确的小数位不固定比如: 1.9999999     2.3496800000001需要动态配合探测失准位置,而不是“一律小数点2位四舍五入”这样会导致更大的误差
      

  5.   

    标题     在Java中实现浮点数的精确计算    AYellow(原作) 修改    
    关键字     Java 浮点数 精确计算   
    问题的提出:
    如果我们编译运行下面这个程序会看到什么?
    public class Test{
        public static void main(String args[]){
            System.out.println(0.05+0.01);
            System.out.println(1.0-0.42);
            System.out.println(4.015*100);
            System.out.println(123.3/100);
        }
    };你没有看错!结果确实是
    0.060000000000000005
    0.5800000000000001
    401.49999999999994
    1.2329999999999999Java中的简单浮点数类型float和double不能够进行运算。不光是Java,在其它很多编程语言中也有这样的问题。在大多数情况下,计算的结果是准确的,但是多试几次(可以做一个循环)就可以试出类似上面的错误。现在终于理解为什么要有BCD码了。
    这个问题相当严重,如果你有9.999999999999元,你的计算机是不会认为你可以购买10元的商品的。
    在有的编程语言中提供了专门的货币类型来处理这种情况,但是Java没有。现在让我们看看如何解决这个问题。 四舍五入
    我们的第一个反应是做四舍五入。Math类中的round方法不能设置保留几位小数,我们只能象这样(保留两位):
    public double round(double value){
        return Math.round(value*100)/100.0;
    }非常不幸,上面的代码并不能正常工作,给这个方法传入4.015它将返回4.01而不是4.02,如我们在上面看到的
    4.015*100=401.49999999999994
    因此如果我们要做到精确的四舍五入,不能利用简单类型做任何运算
    java.text.DecimalFormat也不能解决这个问题:
    System.out.println(new java.text.DecimalFormat("0.00").format(4.025));
    输出是4.02 BigDecimal
    在《Effective Java》这本书中也提到这个原则,float和double只能用来做科学计算或者是工程计算,在商业计算中我们要用java.math.BigDecimal。BigDecimal一共有4个够造方法,我们不关心用BigInteger来够造的那两个,那么还有两个,它们是:
    BigDecimal(double val) 
              Translates a double into a BigDecimal. 
    BigDecimal(String val) 
              Translates the String repre sentation of a BigDecimal into a BigDecimal.上面的API简要描述相当的明确,而且通常情况下,上面的那一个使用起来要方便一些。我们可能想都不想就用上了,会有什么问题呢?等到出了问题的时候,才发现上面哪个够造方法的详细说明中有这么一段:
    Note: the results of this constructor can be somewhat unpredictable. One might assume that new BigDecimal(.1) is exactly equal to .1, but it is actually equal to .1000000000000000055511151231257827021181583404541015625. This is so because .1 cannot be represented exactly as a double (or, for that matter, as a binary fraction of any finite length). Thus, the long value that is being passed in to the constructor is not exactly equal to .1, appearances nonwithstanding. 
    The (String) constructor, on the other hand, is perfectly predictable: new BigDecimal(".1") is exactly equal to .1, as one would expect. Therefore, it is generally recommended that the (String) constructor be used in preference to this one.原来我们如果需要精确计算,非要用String来够造BigDecimal不可!在《Effective Java》一书中的例子是用String来够造BigDecimal的,但是书上却没有强调这一点,这也许是一个小小的失误吧。
     解决方案
    现在我们已经可以解决这个问题了,原则是使用BigDecimal并且一定要用String来够造。
    但是想像一下吧,如果我们要做一个加法运算,需要先将两个浮点数转为String,然后够造成BigDecimal,在其中一个上调用add方法,传入另一个作为参数,然后把运算的结果(BigDecimal)再转换为浮点数。你能够忍受这么烦琐的过程吗?下面我们提供一个工具类Arith来简化操作。它提供以下静态方法,包括加减乘除和四舍五入:
    public static double add(double v1,double v2)
    public static double sub(double v1,double v2)
    public static double mul(double v1,double v2)
    public static double div(double v1,double v2)
    public static double div(double v1,double v2,int scale)
    public static double round(double v,int scale)附录
    源文件Arith.java:import java.math.BigDecimal;
    /**
     * 由于Java的简单类型不能够精确的对浮点数进行运算,这个工具类提供精
     * 确的浮点数运算,包括加减乘除和四舍五入。
     */public class Arith{    //默认除法运算精度
        private static final int DEF_DIV_SCALE = 10;
        //这个类不能实例化
        private Arith(){
        } 
        /**
         * 提供精确的加法运算。
         * @param v1 被加数
         * @param v2 加数
         * @return 两个参数的和
         */    public static double add(double v1,double v2){
            BigDecimal b1 = new BigDecimal(Double.toString(v1));
            BigDecimal b2 = new BigDecimal(Double.toString(v2));
            return b1.add(b2).doubleValue();
        }    /**
         * 提供精确的减法运算。
         * @param v1 被减数
         * @param v2 减数
         * @return 两个参数的差
         */    public static double sub(double v1,double v2){
            BigDecimal b1 = new BigDecimal(Double.toString(v1));
            BigDecimal b2 = new BigDecimal(Double.toString(v2));
            return b1.subtract(b2).doubleValue();
        }     /**
         * 提供精确的乘法运算。
         * @param v1 被乘数
         * @param v2 乘数
         * @return 两个参数的积
         */    public static double mul(double v1,double v2){
            BigDecimal b1 = new BigDecimal(Double.toString(v1));
            BigDecimal b2 = new BigDecimal(Double.toString(v2));
            return b1.multiply(b2).doubleValue();
        }     /**
         * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到
         * 小数点以后10位,以后的数字四舍五入。
         * @param v1 被除数
         * @param v2 除数
         * @return 两个参数的商
         */    public static double div(double v1,double v2){
            return div(v1,v2,DEF_DIV_SCALE);
        }     /**
         * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指
         * 定精度,以后的数字四舍五入。
         * @param v1 被除数
         * @param v2 除数
         * @param scale 表示表示需要精确到小数点以后几位。
         * @return 两个参数的商
         */    public static double div(double v1,double v2,int scale){
            if(scale<0){
                throw new IllegalArgumentException(
                    "The scale must be a positive integer or zero");
            }
            BigDecimal b1 = new BigDecimal(Double.toString(v1));
            BigDecimal b2 = new BigDecimal(Double.toString(v2));
            return b1.divide(b2,scale,BigDecimal.ROUND_HALF_UP).doubleValue();
        }     /**
         * 提供精确的小数位四舍五入处理。
         * @param v 需要四舍五入的数字
         * @param scale 小数点后保留几位
         * @return 四舍五入后的结果
         */    public static double round(double v,int scale){
            if(scale<0){
                throw new IllegalArgumentException(
                    "The scale must be a positive integer or zero");
            }
            BigDecimal b = new BigDecimal(Double.toString(v));
            BigDecimal one = new BigDecimal("1");
            return b.divide(one,scale,BigDecimal.ROUND_HALF_UP).doubleValue();
        }
    }; 
     
      

  6.   

    文章在
    http://www.csdn.net/develop/read_article.asp?id=17400