比较双倍与零

hat*_*enn 18 java double

我是Java的新手,我一直在尝试实现一种算法,用于查找立方体方程的根.当我计算判别式并尝试检查其相对于零的位置时,就会出现问题.

如果运行它并输入数字"1 -5 8 -4",输出如下:

1 -5 8 -4
p=-0.333333, q=0.074074
disc1=0.001372, disc2=-0.001372
discriminant=0.00000000000000001236
Discriminant is greater than zero.
Run Code Online (Sandbox Code Playgroud)

我知道问题出现了,因为双打的计算并不准确.通常判别式应为0,但最终为0.00000000000000001236.

我的问题是,避免这种情况的最佳方法是什么?我应该检查数字是否介于epsilon邻域为零之间?还是有更好更精确的方法?

提前感谢您的回答.

import java.util.Scanner;

class Cubical {
    public static void main(String[] args) {
        // Declare the variables.
        double a, b, c, d, p, q, gamma, discriminant;

        Scanner userInput = new Scanner(System.in);
        a = userInput.nextDouble();
        b = userInput.nextDouble();
        c = userInput.nextDouble();     
        d = userInput.nextDouble();

        // Calculate p and q.
        p = (3*a*c - b*b) / (3*a*a);
        q = (2*b*b*b) / (27*a*a*a) - (b*c) / (3*a*a) + d/a;

        // Calculate the discriminant.
        discriminant = (q/2)*(q/2) + (p/3)*(p/3)*(p/3);

        // Just to see the values.
        System.out.printf("p=%f, q=%f\ndisc1=%f, disc2=%f\ndiscriminant=%.20f\n", p, q, (q/2)*(q/2), (p/3)*(p/3)*(p/3), (q/2)*(q/2) + (p/3)*(p/3)*(p/3));

        if (discriminant > 0) {
            System.out.println("Discriminant is greater than zero.");
        }
        if (discriminant == 0) {
            System.out.println("Discriminant is equal to zero.");
        }
        if (discriminant < 0) {
            System.out.println("Discriminant is less than zero.");
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

Pet*_*rey 17

最简单的epsilon检查是

if(Math.abs(value) < ERROR)
Run Code Online (Sandbox Code Playgroud)

更复杂的一个与价值成正比

if(Math.abs(value) < ERROR_FACTOR * Math.max(Math.abs(a), Math.abs(b)))
Run Code Online (Sandbox Code Playgroud)

在您的具体情况下,您可以:

if (discriminant > ERROR) {
    System.out.println("Discriminant is greater than zero.");
} else if (discriminant < -ERROR) {
    System.out.println("Discriminant is less than zero.");
} else {
    System.out.println("Discriminant is equal to zero.");
}
Run Code Online (Sandbox Code Playgroud)


Jen*_*der 14

我应该检查数字是否介于epsilon邻域为零之间?

究竟

  • +1必须尊重一个单词,不合格(没有'ifs或buts')回答直接问题. (3认同)