ZetCode

PHP Floating-Point Numbers

last modified April 1, 2025

PHP uses a platform-dependent floating-point type based on IEEE 754, typically 64-bit doubles. This guide explores its properties, precision challenges, and best practices for effective numerical operations in PHP applications.

PHP Float Overview

This section introduces PHP's float type, a key numeric type for decimal values. It follows IEEE 754, usually as a 64-bit double, depending on the system. Understanding its range and precision is vital for accurate computations. The example below demonstrates basic usage and its limits in PHP. This knowledge helps developers manage floating-point quirks effectively.

floatOverview.php
<?php

// Float declarations
$singleFloat = 1.2345678901234567;
$smallFloat = 5.0E-324;
$largeFloat = 1.7976931348623157E308;

echo "Float: $singleFloat\n";
echo "Smallest positive: $smallFloat\n";
echo "Largest: $largeFloat\n";

echo "\nPHP_FLOAT_MIN: " . PHP_FLOAT_MIN . "\n";
echo "PHP_FLOAT_MAX: " . PHP_FLOAT_MAX . "\n";
echo "Size in bytes: " . 8 . "\n"; // Typically 64-bit

// Additional example: Scientific notation
$sciFloat = 1.23E-10;
echo "\nScientific notation: $sciFloat\n";

PHP's float type is a floating-point number adhering to IEEE 754, typically a 64-bit double on modern systems, though its size can vary by platform. It provides 15-17 significant decimal digits and a range from ±5.0×10⁻³²⁴ (codePHP_FLOAT_MIN) to ±1.7976931348623157×10³⁰⁸ (PHP_FLOAT_MAX), stored in 8 bytes usually.

Unlike languages with multiple numeric types, PHP uses float for all non-integer numbers, with no distinct decimal type, balancing simplicity and flexibility. This means precision issues, like those with 0.1 + 0.2, are common and need careful handling. Scientific notation (e.g., 1.23E-10) is supported, making it easy to work with very small or large values.

Precision and Rounding

Precision in PHP floats can lead to surprises due to their binary representation. This section shows how decimal fractions cause errors and how PHP mitigates them. Examples use built-in functions to control rounding effectively. Mastering these tools ensures reliable numeric results in PHP scripts. It's essential for avoiding subtle calculation bugs.

precisionExamples.php
<?php

// Precision demonstration
$a = 0.1;
$b = 0.2;
$sum = $a + $b;

echo "0.1 + 0.2 = $sum\n";
echo "0.1 + 0.2 == 0.3? " . ($sum == 0.3 ? 'true' : 'false') . "\n";
echo "Full precision: " . sprintf("%.17f", $sum) . "\n";

$value = 2.34567;
echo "\nRounding examples:\n";
echo "round($value): " . round($value) . "\n";
echo "floor($value): " . floor($value) . "\n";
echo "ceil($value): " . ceil($value) . "\n";
echo "round to 2 decimals: " . round($value, 2) . "\n";

// Additional example: Pi rounding
$pi = 3.14159;
echo "\nPi to 3 digits: " . round($pi, 3) . "\n";

PHP floats, being binary-based per IEEE 754, can't precisely represent decimals like 0.1 or0.2, so 0.1 + 0.2 equals 0.30000000000000004, not 0.3, as seen with sprintf("%.17f"). These errors can compound in loops or repeated math, requiring careful precision management in code.

PHP offers round for nearest value (with optional precision), floor to round down, and ceil to round up, all returning floats. For instance, round(2.34567, 2) gives 2.35, ideal for display or controlled precision, unlike string-based alternatives in other languages. Adjust the php.ini precision setting (default 14) or use formatting to control output as needed.

Comparing Floating-Point Values

Comparing floats in PHP demands caution due to inherent precision inaccuracies. Direct equality checks often fail, so this section provides a safer comparison method. It also addresses special values like NaN and infinity in PHP's context. The example clarifies these issues with practical PHP solutions. This ensures dependable comparison logic in your scripts.

floatComparison.php
<?php

function nearlyEqual($a, $b, $epsilon = 1E-10) {
    $absA = abs($a);
    $absB = abs($b);
    $diff = abs($a - $b);

    if ($a == $b) return true;
    if ($a == 0 || $b == 0 || $diff < PHP_FLOAT_EPSILON)
    {
        return $diff < ($epsilon * PHP_FLOAT_EPSILON);
    }

    return $diff / ($absA + $absB) < $epsilon;
}

$x = 0.1 + 0.2;
$y = 0.3;

echo "Direct equality: " . ($x == $y ? 'true' : 'false') . "\n";
echo "NearlyEqual: " . (nearlyEqual($x, $y) ? 'true' : 'false') . "\n";
echo "Difference: " . ($x - $y) . "\n";

$nan = NAN;
$inf = INF;

echo "\nNAN == NAN: " . ($nan == $nan ? 'true' : 'false') . "\n";
echo "is_nan($nan): " . (is_nan($nan) ? 'true' : 'false') . "\n";
echo "INF == INF: " . ($inf == $inf ? 'true' : 'false') . "\n";

// Additional example: Small number comparison
$small = 1E-15;
echo "\nNearlyEqual($small, 0): " . (nearlyEqual($small, 0) ? 'true' : 'false') . "\n";

In PHP, using == for float comparisons is unreliable, as 0.1 + 0.2 != 0.3 due to tiny binary precision errors, often off by a small fraction like 5.5E-17. The nearlyEqual function compares floats within an epsilon (e.g., 1E-10), using PHP_FLOAT_EPSILON for near-zero cases, ensuring robust checks.

Special values like NAN and INF need care—NAN != NAN, so is_nan is required, while INF == INF holds true but requires context handling. For small numbers like 1E-15 versus 0, nearlyEqual succeeds where == fails, making it a practical solution. This method adapts to PHP's float behavior, enhancing comparison accuracy across applications.

Floating-Point for Financial Calculations

PHP's float type struggles with financial precision due to its binary nature. This section contrasts float issues with a precise integer-based workaround. Examples show interest calculations and monetary rounding in PHP effectively. These techniques ensure accuracy for financial tasks in PHP projects. For complex needs, consider extensions like BCMath instead.

financialExamples.php
<?php

$principal = 1000.00;
$interestRate = 0.05; // 5%
$years = 10;

// Using floating-point
$futureValueFloat = $principal * pow(1 + $interestRate, $years);
echo "Future value (floating-point): $" . round($futureValueFloat, 2) . "\n";

// Workaround with integer cents
$principalCents = 100000; // $1000.00 in cents
$futureValueCents = round($principalCents * pow(1 + $interestRate, $years));
$futureValue = $futureValueCents / 100;
echo "Accurate future value: $" . number_format($futureValue, 2) . "\n";

$payment = 123.456789;
echo "\nPayment to cents: " . round($payment, 2) . "\n";
echo "Payment rounded up: " . ceil($payment * 100) / 100 . "\n";
echo "Payment rounded down: " . floor($payment * 100) / 100 . "\n";

// Additional example: Tax calculation
$price = 19.99;
$taxRate = 0.08;
$tax = round($price * $taxRate, 2);
echo "\nTax on $$price: $$tax\n";

PHP's float type introduces errors in financial math, like compound interest on $1000 at 5% over 10 years, where tiny inaccuracies can appear in the result. Using cents as integers (e.g., 100000 for $1000) and converting back after calculation yields a precise $1628.89, avoiding float drift.

Functions like round($payment, 2), ceil, and floor with cents (e.g., 12345 cents to $123.45) ensure exact monetary values, vital for accounting integrity. A tax on $19.99 at 8% becomes exactly $1.60 with round, aligning with real-world expectations. For high-precision needs, PHP's BCMath extension offers an alternative to floats entirely.

Best Practices

Use PHP floats for general math but switch to integer cents or BCMath for financial accuracy. Avoid == for float comparisons, using nearlyEqual with epsilon instead for reliability. Apply round, ceil, or floor explicitly, and use number_format for clean output formatting. Check NAN with is_nan and INF with is_infinite to handle edge cases properly. These practices keep your PHP numeric code robust and error-free.

Source References

Learn more from these resources: PHP Float Documentation, PHP Math Functions, and BCMath Documentation.

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 PHP tutorials.