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