php integer and float comparison mismatch

I have the following code

$amount1 = 7299;
$amount2 = 72.9875;

$amount2_in_cents = round($amount2, 2) * 100;

if ($amount1 != $amount2_in_cents) {
    echo "Amount $amount1 != $amount2_in_cents\n";

    var_dump($amount1);
    var_dump($amount2_in_cents);    

} else {
    echo "Amounts matched";
}

and this is the output

Amount 7299 != 7299
int(7299)
float(7299)

Now I realise that floats and int are different, but given the rounding i would have expected the two values to match. And I have solved it by casting to int.

So my question is why does this comparison not work as i would have expected (both values matching)?


Solution 1:

Notice the big red warning in the PHP Manual!

Never expect anything when comparing floats. The result of round, even if the precision is 0, is still a float. In your particular case it happened that the result was a little bigger than expected, so casting to int resulted in equality, but for other numbers it might as well happen for it to be a little smaller than expected and casting to int won't round it, but truncate it, so you can't use casting as a workaround. (As a note, a better solution than yours would be casting to string :), but still a lousy option.)

If you need to work with amounts of money always use the BC Math extension.

For rounding with BC Math you can use this technique:

$x = '211.9452';
$x = bcadd($x, '0.005', 2);

Good luck,
Alin

Solution 2:

Use

round()

$float_val = 4.5;
$float_val = round($float_val);

now Compare

Solution 3:

For example a nasty case: We want to check if our ratting is greater or equal than 3.3 out of 5 while in a loop. ( This is all weird example to show a "flaw")

$a="3.3";
for($i=0; $i<5 ; $i=$i+0.15){
    echo "\nTesting if $i>=$a\n";
    var_dump($i,$a);
    if($i>=$a){ 
        echo "$i>=$a is TRUE\n";
    }else{
        echo "$i>=$a is FALSE\n";

    }
}

Now the output will be this one:

Testing if 0>=3.3
int(0)
string(3) "3.3"
0>=3.3 is FALSE

Testing if 0.15>=3.3
float(0.15)
string(3) "3.3"
0.15>=3.3 is FALSE

Testing if 0.3>=3.3
float(0.3)
string(3) "3.3"
0.3>=3.3 is FALSE

Testing if 0.45>=3.3
float(0.45)
string(3) "3.3"
0.45>=3.3 is FALSE

Testing if 0.6>=3.3
float(0.6)
string(3) "3.3"
0.6>=3.3 is FALSE

Testing if 0.75>=3.3
float(0.75)
string(3) "3.3"
0.75>=3.3 is FALSE

Testing if 0.9>=3.3
float(0.9)
string(3) "3.3"
0.9>=3.3 is FALSE

Testing if 1.05>=3.3
float(1.05)
string(3) "3.3"
1.05>=3.3 is FALSE

Testing if 1.2>=3.3
float(1.2)
string(3) "3.3"
1.2>=3.3 is FALSE

Testing if 1.35>=3.3
float(1.35)
string(3) "3.3"
1.35>=3.3 is FALSE

Testing if 1.5>=3.3
float(1.5)
string(3) "3.3"
1.5>=3.3 is FALSE

Testing if 1.65>=3.3
float(1.65)
string(3) "3.3"
1.65>=3.3 is FALSE

Testing if 1.8>=3.3
float(1.8)
string(3) "3.3"
1.8>=3.3 is FALSE

Testing if 1.95>=3.3
float(1.95)
string(3) "3.3"
1.95>=3.3 is FALSE

Testing if 2.1>=3.3
float(2.1)
string(3) "3.3"
2.1>=3.3 is FALSE

Testing if 2.25>=3.3
float(2.25)
string(3) "3.3"
2.25>=3.3 is FALSE

Testing if 2.4>=3.3
float(2.4)
string(3) "3.3"
2.4>=3.3 is FALSE

Testing if 2.55>=3.3
float(2.55)
string(3) "3.3"
2.55>=3.3 is FALSE

Testing if 2.7>=3.3
float(2.7)
string(3) "3.3"
2.7>=3.3 is FALSE

Testing if 2.85>=3.3
float(2.85)
string(3) "3.3"
2.85>=3.3 is FALSE

Testing if 3>=3.3
float(3)
string(3) "3.3"
3>=3.3 is FALSE

Testing if 3.15>=3.3
float(3.15)
string(3) "3.3"
3.15>=3.3 is FALSE

Testing if 3.3>=3.3
float(3.3)
string(3) "3.3"
3.3>=3.3 is FALSE

Testing if 3.45>=3.3
float(3.45)
string(3) "3.3"
3.45>=3.3 is TRUE

Testing if 3.6>=3.3
float(3.6)
string(3) "3.3"
3.6>=3.3 is TRUE

Testing if 3.75>=3.3
float(3.75)
string(3) "3.3"
3.75>=3.3 is TRUE

Testing if 3.9>=3.3
float(3.9)
string(3) "3.3"
3.9>=3.3 is TRUE

Testing if 4.05>=3.3
float(4.05)
string(3) "3.3"
4.05>=3.3 is TRUE

Testing if 4.2>=3.3
float(4.2)
string(3) "3.3"
4.2>=3.3 is TRUE

Testing if 4.35>=3.3
float(4.35)
string(3) "3.3"
4.35>=3.3 is TRUE

Testing if 4.5>=3.3
float(4.5)
string(3) "3.3"
4.5>=3.3 is TRUE

Testing if 4.65>=3.3
float(4.65)
string(3) "3.3"
4.65>=3.3 is TRUE

Testing if 4.8>=3.3
float(4.8)
string(3) "3.3"
4.8>=3.3 is TRUE

Testing if 4.95>=3.3
float(4.95)
string(3) "3.3"
4.95>=3.3 is TRUE

And the nasty part:

 Testing if 3.3>=3.3
    float(3.3)
    string(3) "3.3"
    3.3>=3.3 is FALSE

3.3 is greater or equal than 3.3 but php things not! Makes no sense right

Now if you put a ini_set('precision', 18); before the code you can see that the evaluation actually was:

Testing if 3.29999999999999893>=3.3
float(3.29999999999999893)
string(3) "3.3"
3.29999999999999893>=3.3 is FALSE

So the $i=$i+=0.15 implicitly converts the $i to a float, which will cause this issue.

For this case the $i+=0.15 should be changed to $i=number_format($i+=0.15, 2)