PHP / MySQLi使用预准备语句返回错误的浮点值

时间:2012-07-18 16:01:01

标签: php mysqli prepared-statement php-5.3

行。我可能在这里很厚(已经知道了)但是我找到了一个没有文件记录的特征"在PHP 5.3.x中的MySQLi中准备好的语句,或者我错过了一些非常基础的东西。

短版 - 通过MySQLi和mysqlnd驱动程序在PHP中准备语句返回错误的浮点值

长版

首先,一些测试数据

mysql> CREATE TABLE `t2` (`v` float(6,2) NOT NULL DEFAULT '0.00');
Query OK, 0 rows affected (0.08 sec)

mysql> insert into t2 (v) values (0.03);
Query OK, 1 row affected (0.00 sec)

通过准备好的声明检索上述值时出现问题。使用旧版mysql _...或标准mysqli ...查询调用时,将返回正确的数据。但是,通过预准备语句使用相同的查询会返回不正确的值:

测试代码:

atom:~/testScripts> cat f1.php
<?php
require('passwds.php');

$q="select * from t2"; // same query for all

echo "/* Old style MySQL statements (deprecated) */".PHP_EOL;

$h1=mysql_connect(MYSQL_SERVER,MYSQL_USER,MYSQL_PASSWORD);
mysql_select_db('test',$h1);
$r1=mysql_query($q);
while ($f1=mysql_fetch_assoc($r1))
{
    echo print_r($f1,true).PHP_EOL;
}

echo "/* New style MySQLi statements */".PHP_EOL;

$h2=new mysqli(MYSQL_SERVER,MYSQL_USER,MYSQL_PASSWORD,'test');
$r2=$h2->query($q);
while ($f2=$r2->fetch_assoc())
{
    echo print_r($f2,true).PHP_EOL;
}

echo "/* New style MySQLi prepared statements */".PHP_EOL;

$h3=new mysqli(MYSQL_SERVER,MYSQL_USER,MYSQL_PASSWORD,'test');
$s3=$h3->stmt_init();
$s3->prepare($q);
$s3->execute(); // no binding required
$r3=$s3->get_result();
while ($f3=$r3->fetch_assoc())
{
    echo print_r($f3,true).PHP_EOL;
}

结果:

atom:~/testScripts> php -f f1.php
/* Old style MySQL statements (deprecated) */
Array
(
    [v] => 0.03
)

/* New style MySQLi statements */
Array
(
    [v] => 0.03
)

/* New style MySQLi prepared statements */
Array
(
    [v] => 0.029999999329448
)

如果我将表vt2的类型更改为DOUBLE而不是FLOAT,则预准备语句返回的值是正确的。

我在Linux机器上运行脚本(OpenSuse 12.1 32位,MySQL和PHP从源代码编译)和Windows 7(64位,其中PHP是从带有来自Linux机器的数据的二进制文件安装)时得到相同的响应。我也尝试过OpenSuse 12.1 64bit安装(再次,使用mysqlnd驱动程序从源代码安装PHP和MySQL),结果相同。所有版本都使用PHP提供的mysqlnd驱动程序

问题我是否遗漏了一些非常重要的内容,或者我是否应该向PHP.NET报告错误?

对不起,这是一个很长的问题,但我想我会提供尽可能多的数据

1 个答案:

答案 0 :(得分:2)

实际上旧的(前两个)陈述是错误的。浮点值本质上是不精确的,例如,没有办法以浮点格式精确表示许多精确的小数。因此,.03最接近的浮点值是.0299999 ...某事。您的前两个语句正在享受MySQL驱动程序和/或PHP在数据类型转换期间为您舍入值的“好处”。但是,MySQLi预处理语句使用二进制结果格式,将值“原样”直接转换为PHP数据类型,如int,float,string等。(http://php.net/manual/en/mysqli.quickstart.prepared-statements.php

对浮点精度问题(例如http://dev.mysql.com/doc/refman/5.5/en/problems-with-float.html)进行更多研究,看看它们是否真的符合您的需求。如果您想要精确的小数并且愿意牺牲一点存储空间和/或计算速度,请尝试MySQL DECIMAL(X,Y)类型。但是,MySQLi仍然可以在内部将其转换为PHP float / double。