关于PHP转换超过2038年日期出错的问题解决


Posted in PHP onJune 28, 2017

前言

最近在写一个项目接口。测试中发现服务器上测试正常的功能,在本地一直有问题。一步步的排查,最终锁定问题是由于函数strtotime返回了一个false值,导致数据插入数据库失败。

相同代码运行结果不一样,原因那就是环境不一致导致。要么是PHP版本不同,要么是位数不同。

我电脑是64位的。这里是PHP位数不一致,服务器使用64位,而我本地是32位。而strtotime被传入了一个字符串2050-1-1 23:59:59,该参数大于了2038-1-19 03:14:07所以在32位PHP下直接返回false,而64位PHP不受影响。

Y2K38漏洞

导致上述问题的根本原因就是Y2K38漏洞,也被称为Unix Millennium Bug

32位系统或PHP

此漏洞将会影响到所有 32 位系统下用UNIX 时间戳整数来记录时间的 PHP,及其它编程语言。一个整型的变量所能保存的最大时间为 2038 年01月19 日 03:14:07。超过这个时间后,整型数值将会溢出。

64位系统或PHP

64位系统下可以保存的日期最远日期是现在宇宙年龄的21倍——292亿年。所以不会受到该漏洞影响。

如何检测

如何知道你的系统是否收到该漏洞的影响。很简单,直接使用strtotime去转换一个大于2038年1月19日03:14:07日期。或者使用date函数将一个大于2147454847时间戳转换为日期。

下面具体演示一下

方法一

echo date("Y-m-d H:i:s",2556115199);

上面结果如果返回2050-12-31 23:59:59那么就没有问题。如果返回1914-11-25 09:31:43那么就受收到影响。

方法二

var_dump(strtotime("2050-12-31 23:59:59"));

上面结果如果返回2556115199那么就正常。如果返回false那么也会受到影响。

解决方案

方案一

更换系统和PHP均为64位。这个代价比较大,但是可以永久解决问题。

方案二

PHP5.2版本之后提供了一个函数DateTime可以临时解决一下问题。

// 1、日期字符串转换为时间戳
$obj = new DateTime("2050-12-31 23:59:59");
echo $obj->format("U"); // 2556115199

// 2、时间戳转换为日期字符串
$obj = new DateTime("@2556115199"); // 这里时间戳前要写一个@符号
$timezone = timezone_open('Asia/HONG_KONG'); // 设置时区
$obj->setTimezone($timezone); 
echo $obj->format("Y-m-d H:i:s"); // 2050-12-31 23:59:59

// 而且DateTime还可以有其他玩法
$obj = new DateTime("2050-12-31 23:59:59");
echo $obj->format("Y/m/d H:i:s"); // 换种方式输入时间字符串2050/12/31 23:59:59

通过DateTime类来操作日期不会受到Y2K38漏洞的影响,可以最远支持到9999年12月31日

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
实时抓取YAHOO股票报价的代码
Oct 09 PHP
PHP 导出数据到淘宝助手CSV的方法分享
Feb 27 PHP
php json_encode奇怪问题说明
Sep 27 PHP
洪恩在线成语词典小偷程序php版
Apr 20 PHP
ThinkPHP处理Ajax返回的方法
Nov 22 PHP
php去除html标记的原生函数详解
Jan 27 PHP
简单介绍win7下搭建apache+php+mysql开发环境
Aug 06 PHP
php设置页面超时时间解决方法
Sep 22 PHP
PHP实现简单实用的分页类代码
Apr 08 PHP
thinkPHP框架可添加js事件的分页类customPage.class.php完整实例
Mar 16 PHP
PHP实现将几张照片拼接到一起的合成图片功能【便于整体打印输出】
Nov 14 PHP
利用PHP如何统计Nginx日志的User Agent数据
Mar 06 PHP
YII2自动登录Cookie总是失效的解决方法
Jun 28 #PHP
php的常量和变量实例详解
Jun 27 #PHP
Zend Framework框架中实现Ajax的方法示例
Jun 27 #PHP
PHP用continue跳过本次循环中剩余代码的注意点
Jun 27 #PHP
PHP API接口必备之输出json格式数据示例代码
Jun 27 #PHP
利用PHP获取访客IP、地区位置、浏览器及来源页面等信息
Jun 27 #PHP
简单谈谈PHP面向对象之标识对象
Jun 27 #PHP
You might like
linux php mysql数据库备份实现代码
2009/03/10 PHP
ThinkPHP 防止表单重复提交的方法
2011/08/08 PHP
php的闭包(Closure)匿名函数详解
2015/02/22 PHP
Laravel 中使用 Vue.js 实现基于 Ajax 的表单提交错误验证操作
2017/06/30 PHP
Laravel使用Queue队列的技巧汇总
2019/09/02 PHP
prototype与jquery下Ajax实现的差别
2009/09/13 Javascript
扩展easyui.datagrid,添加数据loading遮罩效果代码
2010/11/02 Javascript
javascript对下拉列表框(select)的操作实例讲解
2013/11/29 Javascript
node.js中的fs.realpathSync方法使用说明
2014/12/16 Javascript
简单的jQuery入门指引
2015/07/28 Javascript
JS实现用户注册时获取短信验证码和倒计时功能
2016/10/27 Javascript
在node.js中怎么屏蔽掉favicon.ico的请求
2017/03/01 Javascript
React Native第三方平台分享的实例(Android,IOS双平台)
2017/08/04 Javascript
微信小程序如何使用云开发
2019/05/17 Javascript
[02:10]2018DOTA2亚洲邀请赛赛前采访-Liquid
2018/04/03 DOTA
[01:15:16]DOTA2-DPC中国联赛 正赛 Elephant vs Aster BO3 第一场 1月26日
2021/03/11 DOTA
Python使用email模块对邮件进行编码和解码的实例教程
2016/07/01 Python
Python实现八大排序算法
2016/08/13 Python
Python实现的栈(Stack)
2018/01/26 Python
我用Python抓取了7000 多本电子书案例详解
2019/03/25 Python
PyQt5响应回车事件的方法
2019/06/25 Python
Django分页功能的实现代码详解
2019/07/29 Python
python实现人机猜拳小游戏
2020/02/03 Python
关于Tensorflow使用CPU报错的解决方式
2020/02/05 Python
python中元组的用法整理
2020/06/15 Python
购买限量版收藏品、珠宝和礼品:Bradford Exchange
2016/09/23 全球购物
NBA德国官方网上商店:NBA Store德国
2018/04/13 全球购物
JDO的含义
2012/11/17 面试题
产品质量承诺书范文
2014/03/27 职场文书
2014年团支部工作总结
2014/11/17 职场文书
2015年学生会部门工作总结
2015/04/21 职场文书
培训后的感想
2015/08/07 职场文书
新学期开学寄语2016
2015/12/04 职场文书
python实现web邮箱扫描的示例(附源码)
2021/03/30 Python
MySQL下使用Inplace和Online方式创建索引的教程
2021/05/26 MySQL
mysql函数全面总结
2021/11/11 MySQL