ZetCode

Java Floating-Point Numbers

last modified April 1, 2025

Java offers two floating-point types: float (32-bit) and double (64-bit), both IEEE 754-based. This guide examines their traits, precision nuances, and best practices for numerical tasks.

Java Float and Double Overview

This section explores Java's two floating-point types: float and double. Both adhere to IEEE 754, differing in size and precision for varied use cases. Grasping their ranges and limits is key to choosing wisely. The example below highlights their properties with sample values. This insight aids in crafting efficient Java programs.

FloatOverview.java
void main() {

    // Float and double declarations
    float simpleFloat = 3.14159f;
    double preciseDouble = 3.141592653589793;
    
    System.out.println("Float: " + simpleFloat);
    System.out.println("Double: " + preciseDouble);
    
    // Range constants
    System.out.println("\nFloat MIN: " + Float.MIN_VALUE);
    System.out.println("Float MAX: " + Float.MAX_VALUE);
    System.out.println("Double MIN: " + Double.MIN_VALUE);
    System.out.println("Double MAX: " + Double.MAX_VALUE);
    
    // Size in bytes
    System.out.println("\nFloat size: " + Float.BYTES + " bytes");
    System.out.println("Double size: " + Double.BYTES + " bytes");

    // Additional example: Exponential notation
    double expDouble = 2.5e-12;
    System.out.println("\nExponential double: " + expDouble);
}

Java's float type is a 32-bit IEEE 754 single-precision number, holding 6-7 significant digits with a range of ±1.4×10⁻⁴⁵ to ±3.4×10³⁸, marked by an ‘f' suffix. The double type, a 64-bit double-precision variant, offers 15-16 digits and spans ±4.9×10⁻³²⁴ to ±1.7×10³⁰⁸, serving as Java's default for decimals.

These types, stored in 4 and 8 bytes respectively, cater to different needs—float for memory efficiency, double for accuracy. Exponential notation (e.g., 2.5e-12) works with both, simplifying work with extreme values. Choose float for constrained memory scenarios and double for precision-critical tasks.

Precision and Rounding

Java's floating-point types exhibit precision quirks due to binary storage. This section reveals how small decimals misalign and how to correct them. Examples showcase Java's rounding tools for practical control. Understanding these behaviors prevents unexpected outcomes in calculations. It's vital for reliable numeric processing in Java.

PrecisionExamples.java
void main() {

    // Precision issue
    double x = 0.3;
    double y = 0.6;
    double result = x + y;
    
    System.out.println("0.3 + 0.6 = " + result);
    System.out.println("0.3 + 0.6 == 0.9? " + (result == 0.9));
    System.out.printf("Full precision: %.17f%n", result);
    
    // Rounding examples
    double num = 5.6789;
    System.out.println("\nRounding examples:");
    System.out.println("Round " + num + ": " + Math.round(num));
    System.out.println("Floor " + num + ": " + Math.floor(num));
    System.out.println("Ceil " + num + ": " + Math.ceil(num));
    
    // Additional example: Formatting
    System.out.printf("Formatted to 2 decimals: %.2f%n", num);
}

Java's float and double types, rooted in IEEE 754, struggle with exact decimal representation, so 0.3 + 0.6 becomes 0.8999999999999999, not 0.9. This stems from binary approximation, where errors can stack up in iterative math, demanding attention.

Java provides Math.round for nearest integer, Math.floor to truncate down, and Math.ceil to bump up, all adjusting double values effectively. For instance, Math.round(5.6789) yields 6, while printf with %.2f formats to 5.68 for display purposes. These tools help manage precision, but developers must anticipate such deviations in design.

Comparing Floating-Point Values

Comparing floats or doubles in Java needs finesse due to precision drift. Simple equality tests often mislead, so this section offers a safer tactic. It also tackles special values like infinity and NaN uniquely in Java. The example demonstrates these techniques with clear outcomes. This ensures trustworthy comparisons in Java codebases.

FloatComparison.java
boolean nearlyEqual(double a, double b, double epsilon) {

    double diff = Math.abs(a - b);
    
    if (a == b) return true;
    
    if (a == 0 || b == 0 || diff < Double.MIN_VALUE) {
        return diff < (epsilon * Double.MIN_VALUE);
    }

    return diff / (Math.abs(a) + Math.abs(b)) < epsilon;
}

void main() {
    double a = 0.3 + 0.6;
    double b = 0.9;
    
    System.out.println("Direct equality: " + (a == b));
    System.out.println("NearlyEqual: " + nearlyEqual(a, b, 1e-10));
    System.out.printf("Difference: %.17f%n", a - b);
    
    // Special values
    double nan = Double.NaN;
    double inf = Double.POSITIVE_INFINITY;
    
    System.out.println("\nNaN == NaN: " + (nan == nan));
    System.out.println("Double.isNaN(nan): " + Double.isNaN(nan));
    System.out.println("INF == INF: " + (inf == inf));
    
    // Additional example: Tiny value check
    double tiny = 1e-14;
    System.out.println("\nNearlyEqual(" + tiny + ", 0): " + nearlyEqual(tiny, 0, 1e-10));
}

In Java, comparing floats or doubles with == is shaky, as 0.3 + 0.6 != 0.9 due to minute binary errors, often differing by fractions like 1.1e-16. The nearlyEqual method uses an epsilon (e.g., 1e-10) to assess closeness, factoring in Double.MIN_VALUE for tiny numbers, offering a solid alternative.

Special cases like Double.NaN (never equal to itself, checked with isNaN) and Double.POSITIVE_INFINITY (equal to itself) require distinct handling for safety. For near-zero values like 1e-14, nearlyEqual excels where == falters, ensuring precision in comparisons. This method aligns with Java's strict typing for dependable results.

Floating-Point for Financial Calculations

Java's float and double types falter in financial contexts needing exact decimals. This section contrasts their flaws with an integer-based fix for accuracy. Examples illustrate interest computation and currency rounding in Java. These approaches guarantee precision for monetary applications. For robust solutions, Java's BigDecimal is advised instead.

FinancialExamples.java
void main() {

    double principal = 2000.00;
    double rate = 0.04; // 4%
    int years = 8;
    
    // Floating-point calculation
    double futureValueFloat = principal * Math.pow(1 + rate, years);
    System.out.printf("Future value (float): $%.2f%n", futureValueFloat);
    
    // Integer cents for accuracy
    long principalCents = 200_000; // $2000.00 in cents
    long futureValueCents = Math.round(principalCents * Math.pow(1 + rate, years));
    double futureValue = futureValueCents / 100.0;
    System.out.printf("Accurate future value: $%.2f%n", futureValue);
    
    // Rounding example
    double amount = 456.7891;
    System.out.println("\nAmount to cents: " + Math.round(amount * 100) / 100.0);
    
    // Additional example: Discount calculation
    double price = 29.95;
    double discount = 0.15; // 15%
    double savings = Math.round(price * discount * 100) / 100.0;
    System.out.printf("Discount on $%.2f: $%.2f%n", price, savings);
}

Java's float and double types introduce slight errors in financial math, like interest on $2000 at 4% over 8 years, due to binary rounding quirks. Using cents as a long (e.g., 200_000 for $2000) and converting post-calculation delivers a precise $2738.03, sidestepping float issues.

Rounding with Math.round on cents (e.g., 45678 cents to $456.78) ensures exact currency values, vital for fiscal integrity. A 15% discount on $29.95 yields exactly $4.49 with this method, avoiding drift seen in direct double use. For ultimate precision, import java.math.BigDecimal to handle decimals without binary approximation.

Best Practices

Favor double over float in Java for most tasks, reserving float for memory-tight cases. Use BigDecimal or cents for financial precision, avoiding float/double directly. Skip == for comparisons, employing nearlyEqual with epsilon for reliability. Leverage Math.round, floor, or ceil deliberately, and check NaN/infinity with isNaN/isInfinite. These steps ensure Java numeric operations are both precise and efficient.

Source References

Learn more from these resources: Java Float Docs, Java Double Docs, and BigDecimal Docs.

Author

My name is Jan Bodnar, and I am a passionate programmer with extensive programming experience. I have been writing programming articles since 2007. To date, I have authored over 1,400 articles and 8 e-books. I possess more than ten years of experience in teaching programming.

List all Java tutorials.