在 Java 中四舍五入?

Rounding in Java?

我的作业如下:

将20对数字(分别是身份证号码和分数)读入两个单独的数组。求平均分。打印一个table,如下所示,每个学生的ID、分数和差异(分数-平均),每行一个学生。如图所示,在 table 的头部打印分数的总和、平均值和计数。将平均值和差值四舍五入到小数点后两位。

我的程序中有一个应该非常简单的错误,但由于某种原因我无法弄清楚。我已经想出了如何做整件事,但出于某种原因,我的减法被关闭了。这是一个示例输出以供参考:

ID  Score   Difference
115 257.0   14.349999999999994
123 253.0   10.349999999999994
116 246.0   3.3499999999999943
113 243.0   0.3499999999999943
112 239.0   -3.6500000000000057
104 239.0   -3.6500000000000057
110 238.0   -4.650000000000006
218 243.0   0.3499999999999943
208 242.0   -0.6500000000000057
222 223.0   -19.650000000000006
223 230.0   -12.650000000000006
213 229.0   -13.650000000000006
207 228.0   -14.650000000000006
203 224.0   -18.650000000000006
305 265.0   22.349999999999994
306 262.0   19.349999999999994
311 256.0   13.349999999999994
325 246.0   3.3499999999999943
321 245.0   2.3499999999999943
323 245.0   2.3499999999999943

问题是当我调用我的程序只打印它计算的平均值时,它是一个很好的四舍五入的 242.65。而且由于 score[k] 显然也是一个整数,我不明白这是怎么回事?您有什么办法可以阐明这里出了什么问题吗?

import java.util.Scanner;
import java.io.*;

public class prog402a
{
    public static void main(String args[])
    {
        Scanner inFile = null;
        try
        {
            inFile = new Scanner(new File("prog402a.dat"));
        }
        catch (FileNotFoundException e)
        {
            System.out.println("File not found!");
            System.exit(0);
        }

        int[] idNumber = new int[20];
        double[] score = new double[20];

        for(int k = 0; k < idNumber.length; k++)
        {
            idNumber[k] = inFile.nextInt();
            score[k] = inFile.nextDouble();
        }

        double sum = 0;

        for(int k = 0; k < score.length; k++)
        {
            sum += score[k];
        }

        double doubSum = (double)sum;
        double average = doubSum/20.0;
        average=(int)(average*100+.5)/100.0;
        System.out.println(average);

        System.out.println("ID\tScore\tDifference");
        for(int k = 0; k < idNumber.length; k++)
        {
            System.out.println(idNumber[k] + "\t" + score[k] + "\t" + (score[k] - average));
        }



        inFile.close();

    }
}

浮点数(例如 Java 中的浮点数和双精度数)不
编程语言的精度和数学一样 100%。

所以很可能是一个不错的整数(十进制),例如

242.65

在内部(例如在 JVM 中)表示为例如242.64999999997.

我只是粗略地讲到这里,但你明白了。

在打印每个数字之前,只需将其格式化为小数点后第二位,就可以了。让我尝试为您编写一个小示例......这是:

import java.text.DecimalFormat;
import java.text.NumberFormat;

public class Test020 {

    public static void main(String[] args) {

        double d = 13.5 + 0.1 + 0.11;

        System.out.println(d);

        NumberFormat formatter = new DecimalFormat("#0.00");     
        System.out.println(formatter.format(d));

    }

}

浮点数使用二进制系统来表示数字。这意味着一些可以在十进制系统中用有限数量的数字表示的数字不能在二进制系统中表示。例如,无法正确表示 0.2 二进制。因此,该算法将使用它可以表示的 最接近的值。因此它将与 0.199999...-ish 或 0.2000...1-ish 一起工作。请注意十进制系统也有问题,例如表示 1/3 是用 0.333333.... 完成的,所以如果您要用有限的位数进行计算。你总是会犯(小)错误。

但是存在 class - BigDecimal 能够以任意精度计算分数,尽管我认为计算 20 学生的平均分只会有所不同7 左右的数字。

因此我建议简单地格式化数字,例如,它以两位数精确显示结果。你可以这样做:

DecimalFormat myFormatter = new DecimalFormat("0.000"); //the pattern
String output = myFormatter.format(number);
System.out.println(output);

如前所述,问题在于浮点数并非 100% 准确。在您的情况下,最简单的解决方案可能是 String.format;将您的打印行更改为:

 System.out.println(
     String.format("%d\t%.2f\t%.2f", idNumber[k], score[k], score[k] - average));

这将输出四舍五入到小数点后两位的数字。