你应该知道PHP浮点数知识


Posted in PHP onMay 13, 2015

PHP是一种弱类型语言, 这样的特性, 必然要求有无缝透明的隐式类型转换, PHP内部使用zval来保存任意类型的数值, zval的结构如下(5.2为例):

struct _zval_struct {

    /* Variable information */

    zvalue_value value;     /* value */

    zend_uint refcount;

    zend_uchar type;    /* active type */

    zend_uchar is_ref;

};

上面的结构中, 实际保存数值本身的是zvalue_value联合体:
typedef union _zvalue_value {

    long lval;                  /* long value */

    double dval;                /* double value */

    struct {

        char *val;

        int len;

    } str;

    HashTable *ht;              /* hash table value */

    zend_object_value obj;

} zvalue_value;

今天的话题, 我们只关注其中的俩个成员, lval和dval, 我们要意识到, long lval是随着编译器, OS的字长不同而不定长的, 它有可能是32bits或者64bits, 而double dval(双精度)由IEEE 754规定, 是定长的, 一定是64bits.

请记住这一点, 造就了PHP的一些代码的”非平台无关性”. 我们接下来的讨论, 除了特别指明, 都是假设long为64bits

IEEE 754的浮点计数法, 我这里就不引用了, 大家有兴趣的可以自己查看, 关键的一点是, double的尾数采用52位bit来保存, 算上隐藏的1位有效位, 一共是53bits.

在这里, 引出一个很有意思的问题, 我们用c代码举例(假设long为64bits):

    long a = x;

    assert(a == (long)(double)a);

请问, a的取值在什么范围内的时候, 上面的代码可以断言成功?(留在文章最后解答)

现在我们回归正题, PHP在执行一个脚本之前, 首先需要读入脚本, 分析脚本, 这个过程中也包含着, 对脚本中的字面量进行zval化, 比如对于如下脚本:

<?php

$a = 9223372036854775807; //64位有符号数最大值

$b = 9223372036854775808; //最大值+1

var_dump($a);

var_dump($b);

输出:
int(9223372036854775807)

float(9.22337203685E+18)

也就说, PHP在词法分析阶段, 对于一个字面量的数值, 会去判断, 是否超出了当前系统的long的表值范围, 如果不是, 则用lval来保存, zval为IS_LONG, 否则就用dval表示, zval IS_FLOAT.

凡是大于最大的整数值的数值, 我们都要小心, 因为它可能会有精度损失:

<?php

$a = 9223372036854775807;

$b = 9223372036854775808;

 

var_dump($a === ($b - 1));

输出是false.

现在接上开头的讨论, 之前说过, PHP的整数, 可能是32位, 也可能是64位, 那么就决定了, 一些在64位上可以运行正常的代码, 可能会因为隐形的类型转换, 发生精度丢失, 从而造成代码不能正常的运行在32位系统上.

所以, 我们一定要警惕这个临界值, 好在PHP中已经定义了这个临界值:

<?php

    echo PHP_INT_MAX;

 ?>

 

当然, 为了保险起见, 我们应该使用字符串来保存大整数, 并且采用比如bcmath这样的数学函数库来进行计算.

另外, 还有一个关键的配置, 会让我们产生迷惑, 这个配置就是php.precision, 这配置决定了PHP再输出一个float值的时候, 输出多少有效位.

最后, 我们再来回头看上面提出的问题, 也就是一个long的整数, 最大的值是多少, 才能保证转到float以后再转回long不会发生精度丢失?

比如, 对于整数, 我们知道它的二进制表示是, 101, 现在, 让我们右移俩位, 变成1.01, 舍去高位的隐含有效位1, 我们得到在double中存储5的二进制数值为:

0/*符号位*/ 10000000001/*指数位*/ 0100000000000000000000000000000000000000000000000000

5的二进制表示, 丝毫未损的保存在了尾数部分, 这个情况下, 从double转会回long, 不会发生精度丢失.

我们知道double用52位表示尾数, 算上隐含的首位1, 一共是53位精度.. 那么也就可以得出, 如果一个long的整数, 值小于:

2^53 - 1 == 9007199254740991; //牢记, 我们现在假设是64bits的long

那么, 这个整数, 在发生long->double->long的数值转换时, 不会发生精度丢失.
PHP 相关文章推荐
隐藏你的.php文件的实现方法
Mar 19 PHP
PHP stristr() 函数(不区分大小写的字符串查找)
Jun 03 PHP
PHP中读写文件实现代码
Oct 20 PHP
PHP的博客ping服务代码
Feb 04 PHP
用PHP编写和读取XML的几种方式
Jan 12 PHP
PHP连接和操作MySQL数据库基础教程
Sep 29 PHP
php中socket通信机制实例详解
Jan 03 PHP
PHP中把数据库查询结果输出为json格式简单实例
Apr 09 PHP
PHP输出一个等腰三角形的方法
May 12 PHP
PHP实现的AES双向加密解密功能示例【128位】
Sep 03 PHP
redis+php实现微博(二)发布与关注功能详解
Sep 23 PHP
php 文件上传至OSS及删除远程阿里云OSS文件
Jul 04 PHP
PHP浮点数精度问题汇总
May 13 #PHP
PHP生成器简单实例
May 13 #PHP
php实现比较两个字符串日期大小的方法
May 12 #PHP
php使用substr()和strpos()联合查找字符串中某一特定字符的方法
May 12 #PHP
PHP异常处理浅析
May 12 #PHP
php猴子选大王问题解决方法
May 12 #PHP
PHP嵌套输出缓冲代码实例
May 12 #PHP
You might like
php中将数组转成字符串并保存到数据库中的函数代码
2013/09/29 PHP
php自定义hash函数实例
2015/05/05 PHP
PHP实现抓取迅雷VIP账号的方法
2015/07/30 PHP
关于php unset对json_encode的影响详解
2018/11/14 PHP
ExtJS GTGrid 简单用户管理
2009/07/01 Javascript
关于jQuery object and DOM element
2013/04/15 Javascript
JQuery实现的购物车功能(可以减少或者添加商品并自动计算价格)
2015/01/13 Javascript
常用的JavaScript模板引擎介绍
2015/02/28 Javascript
js简单时间比较的方法
2016/08/02 Javascript
在Web项目中引入Jquery插件报错的完美解决方案(图解)
2016/09/19 Javascript
基于JavaScript实现前端文件的断点续传
2016/10/17 Javascript
jQuery中get方法用法分析
2016/12/07 Javascript
vue-resource 拦截器(interceptor)的使用详解
2017/07/04 Javascript
如何用webpack4带你实现一个vue的打包的项目
2018/06/20 Javascript
vue实现动态添加数据滚动条自动滚动到底部的示例代码
2018/07/06 Javascript
JAVA面试题 static关键字详解
2019/07/16 Javascript
Centos7 安装Node.js10以上版本的方法步骤
2019/10/15 Javascript
vue element 中的table动态渲染实现(动态表头)
2019/11/21 Javascript
javascript全局自定义鼠标右键菜单
2020/12/08 Javascript
Python操作列表的常用方法分享
2014/02/13 Python
python实时分析日志的一个小脚本分享
2017/05/07 Python
Python numpy实现数组合并实例(vstack,hstack)
2018/01/09 Python
如何在Anaconda中打开python自带idle
2020/09/21 Python
python3从网络摄像机解析mjpeg http流的示例
2020/11/13 Python
Python爬虫破解登陆哔哩哔哩的方法
2020/11/17 Python
lookfantastic荷兰:在线购买奢华护肤、护发和化妆品
2018/11/27 全球购物
医院护理人员的自我评价分享
2013/10/04 职场文书
个人求职信范例
2014/01/29 职场文书
小学生环保演讲稿
2014/04/25 职场文书
移交协议书
2014/08/19 职场文书
给老师的一封感谢信
2015/01/20 职场文书
爱心捐书倡议书
2015/04/27 职场文书
关于公司年会的开幕词
2016/03/04 职场文书
闭幕词的写作格式与范文!
2019/06/24 职场文书
Python GUI编程之tkinter 关于 ttkbootstrap 的使用详解
2022/03/03 Python
PYTHON 使用 Pandas 删除某列指定值所在的行
2022/04/28 Python