由于php 32位使用 int 類型保存時(shí)間戳,也就是從1970 00:00:00 到當(dāng)前時(shí)間的秒數(shù)。
而32位int 數(shù)字的取值范圍是 -2147483648 到 2147483647。
所以當(dāng) 時(shí)間戳為最大值 2147483647 時(shí),表示的時(shí)間是 2038-01-19 03:14:07 或北京時(shí)間 2038-01-19 11:14:07 (為了表述方便,下文中,將這個(gè)臨界點(diǎn)時(shí)間稱之為 T0)。
而當(dāng)時(shí)間大于這個(gè)時(shí)間時(shí),php很多內(nèi)置函數(shù)都會(huì)出錯(cuò)。
比如
當(dāng)日期和時(shí)間大于 北京時(shí)間 2038-01-19 11:14:07 時(shí)
time()函數(shù),原本應(yīng)該返回時(shí)間戳,現(xiàn)在會(huì)始終返回-1。
date("Y-m-d H:i:s")函數(shù),會(huì)返回 1970-01-01 07:59:59(北京時(shí)間),其實(shí)也是因?yàn)?time()=-1導(dǎo)致的,date默認(rèn)的第二個(gè)參數(shù)就是time()。
同樣,mktime() 等函數(shù)也會(huì)異常。
上網(wǎng)查了解決辦法,
1、換用64位系統(tǒng)。這里說(shuō)的64位系統(tǒng),需要操作系統(tǒng)、web服務(wù)系統(tǒng),以及PHP都要64位的。
2、使用php5.2之后推出的 DateTime 類。
首先說(shuō)第一種方法,因?yàn)槲业姆?wù)器建設(shè)在Windows系統(tǒng)上,然后又有幾個(gè)自制插件,這些插件在php 64位下面可能不能使用,因此這個(gè)方法不能用。
再說(shuō)DateTime類,網(wǎng)上幾乎幾十篇文章都說(shuō)使用DateTime類就能解決2038年問(wèn)題。
我在自己的服務(wù)器上測(cè)試了一下,使用DateTime類似乎確實(shí)可以讓日期超過(guò)2038上限,各種轉(zhuǎn)換,都沒(méi)問(wèn)題,這里我不具體說(shuō)明,大家網(wǎng)上搜"php datetime",都有說(shuō)明。
但是,我在把服務(wù)器的時(shí)間設(shè)置為2040年4月18日的時(shí)候,發(fā)現(xiàn),datetime 類依然無(wú)法獲取當(dāng)前時(shí)間。代碼如下:
$date = new DateTime();
echo $date->format('Y-m-d H:i:s');
輸出的還是 1970-01-01 07:59:59
但是,如果使用 $date->setDate(2040,4,18) 之后,再顯示,再輸出時(shí)間戳等,都是正常的代碼如下:
$date = new DateTime();
$date->setDate(2040,4,18);
$date->setTime(10,24,11);
echo $date->format('Y-m-d H:i:s')."
rn";
echo $date->format('U')."
rn";
這時(shí) 輸出時(shí)間 2040-04-18 10:24:11 ,以及時(shí)間戳 2218328651 都是正常的。
問(wèn)題在于,datetime 類可以解決 2038年之后的時(shí)間的各種運(yùn)算和轉(zhuǎn)換,但是當(dāng)系統(tǒng)日期在2038年那個(gè)T0時(shí)間之后,php系統(tǒng)根本無(wú)法獲取當(dāng)前時(shí)間。
我還試了 new DateTime("today");new DateTime('+2 days');new DateTime('tomorrow'); 等等,都無(wú)法獲取今天,明天,后天等日期。
這時(shí),整個(gè)php 系統(tǒng)無(wú)法獲取當(dāng)前的年月日和時(shí)間。
然后我開始在php的系統(tǒng)數(shù)組 $_SERVER 中尋找,看看哪里能找到和時(shí)間相關(guān)的內(nèi)容,終于被我找到一個(gè) $_SERVER["REQUEST_TIME"],這個(gè)實(shí)際上是一個(gè)記錄用戶刷新頁(yè)面時(shí)php相應(yīng)時(shí)刻的時(shí)間。它的值,在T0之前,和time()是一致的,但是,當(dāng)T0之后,它就變成負(fù)數(shù)了。那么,怎么通過(guò) $_SERVER["REQUEST_TIME"] 來(lái)獲取真實(shí)的 時(shí)間戳呢?
很簡(jiǎn)單,32位int 數(shù)字的取值范圍是 -2147483648 到 2147483647,轉(zhuǎn)成2進(jìn)制就會(huì)發(fā)現(xiàn),其實(shí)是最高位用作符號(hào)位,最高位0表示正數(shù),最高位1表示負(fù)數(shù),當(dāng)數(shù)字達(dá)到 2147483647后,二進(jìn)制 就是 01111111 11111111 11111111 11111111(31個(gè)1),這時(shí)就是T0時(shí)刻的時(shí)間戳,繼續(xù)+1 以后,變成了 10000000000000000000000000000000 (31個(gè)0),如果是無(wú)符號(hào)32位整數(shù),就是 2147483648(正數(shù)) 但是在有符號(hào)的整數(shù)里,最高位1表示負(fù)數(shù),就是 -2147483648(負(fù)數(shù)),而 $_SERVER["REQUEST_TIME"] 的特性是根據(jù)時(shí)間的推移進(jìn)行累加。所以,它的時(shí)間線如下:
T0 之前:它等于 1970 00:00:00 到當(dāng)前時(shí)間的秒數(shù),和time()相同
T0 時(shí): 它等于 2147483647
T0 后1秒: 它等于 2147483647+1=2147483648 被表示為 -2147483648 我們把 -2147483648 記作 T1,T1=T0+1秒的時(shí)刻
T0 后N秒:-2147483648-1+N
所以,當(dāng) $_SERVER["REQUEST_TIME"]<0 時(shí),真正的時(shí)間戳為 $_SERVER["REQUEST_TIME"]-(-2147483648)+ 2147483647。
其中 $_SERVER["REQUEST_TIME"]-(-2147483648)表示 T1(變成負(fù)數(shù),即T0+1秒) 時(shí)刻到當(dāng)前時(shí)間 過(guò)了多少秒。
據(jù)此,寫出一個(gè)新的取代time()的函數(shù),該函數(shù)在系統(tǒng)時(shí)間超過(guò)T0 時(shí),也能返回正確的時(shí)間戳,但是它的范圍是無(wú)符號(hào)32位上限 4294967295,北京時(shí)間 2106-02-07 06:28:15。在這個(gè)時(shí)間之前,應(yīng)該都可以正常使用。
function sunTime(){
if($_SERVER["REQUEST_TIME"]>0){
$t=$_SERVER["REQUEST_TIME"];
}else{
$t0=PHP_INT_MAX; // 第 2147483647 秒 再過(guò)一秒為 2147483648秒,但最高位變成1,系統(tǒng)中為 -2147483648
$t1=0-$t0-1; // t0后面1秒,瞬間變成負(fù)數(shù),值為 -2147483648
$t2=($_SERVER["REQUEST_TIME"]);//雖然$_SERVER["REQUEST_TIME"]變成了負(fù)數(shù),但是 系統(tǒng)依然通過(guò) +1秒 來(lái)計(jì)時(shí)
$t=$t2-$t1+$t0.""; //t2-t1 就是變成負(fù)數(shù)后過(guò)了多少秒,t0就是變成負(fù)數(shù)前的秒數(shù)。
}
$date=new datetime("@".$t);
$timemark=$date->format("U");
return $timemark;
}
所以,目前網(wǎng)上很多人都以為使用 DateTime類可以解決問(wèn)題,殊不知等時(shí)間真正到了2038那個(gè)時(shí)間之后,php系統(tǒng)獲得當(dāng)前時(shí)間都會(huì)出錯(cuò)。而我這方法也是目前網(wǎng)上唯一存在的方法。
陽(yáng)光浪子
2018-04-18