我知道这个问题已经重复很多,但是请先查看该语句,然后再将其标记为已回答:)
为了将双精度值截断为小数点后两位小数,我使用了两种方法,这些方法到处都有提到。它们在下面给出
//this one
DecimalFormat dtime = new DecimalFormat("#.##");
return Double.valueOf(dtime.format(val));
//or the one below
BigDecimal bd = new BigDecimal(val);
BigDecimal rounded = bd.setScale(2, BigDecimal.ROUND_HALF_UP);
return rounded.doubleValue();
问题在于,对于这两种方式,我大多都会在数据集中获得正确的舍入值。但是奇怪的是,同时我得到的值是2.00000000000005或19.97999999999。
我没有得到的问题是,为什么只对一些值进行了四舍五入。有什么问题吗?
为了将双精度值截断为小数点后两位小数,我使用了两种方法,这些方法到处都有提到。
而且他们俩都是错的,因为他们正在尝试不可能的事情。 没有这样的东西会将双精度字符截断为两位小数,因为双精度数不会have小数位。他们有二进制位置。请参见my answer here以作证明。如果要保留小数位,则必须使用十进制基数,即BigDecimal
或DecimalFormat
。
问题是,给定基础表示,浮点数本质上本质上是近似的。因此,您将希望在近似值良好的地方使用它们,而在近似值不好的地方(例如财务状况)避免使用它们。
对rounded.doubleValue()
的调用仍然返回浮点数,因此它仍然受到表示形式的限制的影响。
请参阅
http://docs.oracle.com/cd/E19957-01/806-3568/ncg_goldberg.html
有关更多信息。
下面的代码帮助我限制了双精度值的小数位数(截断)。
public static Double truncate (double valueToTruncate, int numberOfDecimalPlaces) {
if (valueToTruncate > 0) {
return new BigDecimal(String.valueOf(valueToTruncate)).setScale(numberOfDecimalPlaces, BigDecimal.ROUND_FLOOR).doubleValue();
} else {
return new BigDecimal(String.valueOf(valueToTruncate)).setScale(numberOfDecimalPlaces, BigDecimal.ROUND_CEILING).doubleValue();
}
}
希望这对某人有帮助:)
我对此并不陌生,但是让所有事情都在我眼前,我是这样做的。现在请注意,这在数学上是一个截断,除了在每行之后的调试中,我不会将其转换为字符串。
它不优雅,但似乎可以工作。这纯粹是一个解决问题的框架。
反正;
import java.util.Scanner;
public class TestConstructs
{
private static double w;
private static int w1;
private static double w2;
private static double w3;
private static int w4;
private static double w5;
private static double w6;
public static void main(String[] args)
{
// TODO Auto-generated method stub
TestConstructs foo = new TestConstructs();
foo.setWage(w);
}
public void setWage(double w)
{
Scanner input = new Scanner(System.in);
System.out.println("Enter Wage: "); //enter something longish like 30987.978654 or w/e
w = input.nextDouble();
w1 = (int)w;
System.out.printf("%d w1\n",w1);
w2 = w - w1;
System.out.printf("%.3f w2\n",w2);
w3 = w2*100;
System.out.printf("%.3f w3\n",w3);
w4 = (int)w3;
System.out.printf("%d w4\n",w4);
w5 = (double)w4;
System.out.printf("%.3f w5\n",w5);
w6 = 0 + w5/100;
System.out.printf("%.3f w6\n",w6);
w = w1 + w6;
System.out.printf("%.3f final value\n",w); //.3 to show zero
input.close();
}
}
我最后得到了什么输入工资:30987.97865430987 w10.979平方米97.865 w397 w497.000 w50.970 w6最终值30987.970