2009年1月12日 星期一

[PHP] 時間函數運用與範例

本文包含以下內容:
1、 得到目前的日期和時間-我們有多少種方式?
2、 改變日期顯示的方式-日期和時間的顯示形式
3、 轉換現在的日期為Unix的時間戳值
4、 改變日期
a. 增加時間
b. 減去時間
c. 找出兩日期之間的間隔
5、 為PHP添加DateAdd函數
6、 為PHP添加DateDiff函數

**得到目前的日期和時間

在Unix中,時間的表示方式為計算從1970年1月1日零時起所過去的秒數,這稱為UNIX 時間戳(Unix Epoch)。
如果我們有這樣一段的代碼:

echo time();

將返回值958905820
而此時的時間為2000年5月21日12時43分。
你 也許會說這相當不錯。當這對我毫無幫助,或者只有一點幫助。在PHP中,對日期處理的函數都必須用到由time()返回的時間戳值。同時,由於PHP在 Unix和Windows系統中均使用同樣的時間戳值,這就允許你不需要修改代碼即可在不同的系統間移植。另外的一個好處是time()函數返回的是一個 整數,你可以將其作為整數字段或文本字段存入數據庫,而不必使用特別的日期/時間字段。
你已經基本瞭解了Unix的時間戳值,現在讓我們來展示它的實際用途。

改變日期顯示的方式-日期和時間的顯示形式

PHP提供兩個辦法來將Unix的時間戳值轉換成為有用的數據。第一個是date()函數。這個函數有兩個參數-第一個字符串用於設定你所希望返回的格式,第二個為Unix的時間戳值。
格式化字符串通過一些簡單的特殊格式化字符來顯示你所希望看到的格式的日期和時間。假設你希望日期以這樣的格式顯示「18h01 Sunday 21 May」。
我 們需要對字符串中的每一部分使用一個特殊格式化字符,你可以從PHP手冊中日期和時間函數庫中找到。這樣的特殊格式化字符數量不少,他們所表示的類似於星 期幾、月的英文名、用2位或4位數表示的年份,是否是上午(AM)或下午(PM)以及其他。對於這個例子我們需要的特殊字符為:
'H' -24 小時制的小時
'i'- 分鐘
'l'- 星期幾的英文全名
'd'- 本月的第幾日
'F'- 月份的英文全名
因此我們的格式化字符串為」Hhi l d F」, PHP代碼為:

echo date ("Hhi l d F" ,time());

當我們執行這段代碼,我們發現我們所得到的結果為:
180609 Sunday 21 May
這樣的結果看起來有些奇怪。讓我們再查一下PHP手冊,原來』h'所代表的是12 小時制的小時數。這再次證明了一句真理:「計算機只做你所告訴它該做的,而不是你想要它做的」。我們有兩個選擇。第一個是在h前使用轉義字符「\」:

echo date ("H\hi l d F", time());

我們得到這樣的結果:
18h12 Sunday 21 May

這正是我們所要的。但如果我們在一個十分複雜的句子中需要包含日期和時間,我們是否需要對每個字符使用轉義字符?
答案當然是不。我們使用另一個函數strftime()。
strftime() 有兩個好處。第一個好處我們並不在本文討論範圍內-如果你使用setlocale()函數,你可以通過strftime得到相應語言的月份的名稱。另外的 一個好處是你可以將特別的日期和時間的格式化字符包含在你的字符串中。這同時也意味著無論你是否要學習date()函數的所有特殊格式化字符,你都必須學 習一整套完全不同的格式化字符。
strftime()工作的方式和date()沒有什麼不同,除了特殊格式化字符的前面必須添加一個百分號%。如果用strftime()函數,前面例子的代碼如下:

echo strftime ("%Hh%M %A %d %b" ,time());

結果為:
18h24 Sunday 21 May

這也許看起來將簡化繁,但考慮一下如果你所需要的顯示的為"Today is Sunday 21 May 2000. The time is somewhere close to 18h24." 我想使用date()函數無疑令人感到厭煩。
在開始的時候,我提及我們有兩種方式可以從Unix時間戳值中得到有用的數據。我們剛剛瞭解了date()和strftime()。另一個getdate()。這個函數只需要Unix 的時間戳值作為參數,而函數的返回值為日期和時間的數組。

$date_time_array = getdate (time());
echo $date_time_array[ "weekday"];

返回的結果為:
Sunday

除了"weekday",該數組的其他部分為:
"seconds" –秒
"minutes" –分
"hours" –小時
"mday" - 本月的第幾天
"wday" -本週的第幾天(數字)
"mon" -月(數字)
"year" –年
"yday" - r本年的第幾天(數字)
"month" -月份全名
我們現在可以得到容易辨認的日期和時間。那麼其他呢?

**轉換現在的日期為Unix的時間戳值

通常你必須處理一些日期或時間格式的數據。打開M$的一個Access數據庫,所有的日期都以YYYY/MM/DD的格式存儲,加入目前的日前即為2000/05/27。Mktime()函數可以將一個時間轉換成Unix的時間戳值。
函數的格式為:
int mktime( int hour, int minute, int second, int month, int day, int year, int [is_dst] );
從左往右你必須提供小時、分、秒、月、天和年。最後一個參數用於指定你是否處於夏令時,此參數是可選的,所以我們將忽略它。
代碼如下:

echo mktime (0, 0,0 ,5, 27,2000 );

由於不知道小時、分和秒同時這些參數必須填寫,我將其設置為0。設置為0意味著時間為午夜。

$access_date = "2000/05/27";
//explode()函數用一個字符串作為分界來分解另一個字符串。這個例子$access_date通過字符串」/」來分解
$date_elements = explode("/" ,$access_date);
// 此時
// $date_elements[0] = 2000
// $date_elements[1] = 5
// $date_elements[2] = 27
echo mktime (0, 0,0 ,$date_elements [1], $date_elements[ 2],$date_elements [0]);

=============================================================
我們看一個比從Access數據庫單純獲得日期更複雜的情況,我們得到一個以下格式的日期和時間:2000/05/27 02:40:21 PM

// 來自Access的字符串
$date_time_string = "2000/05/27 02:40:21 PM";
// 將字符串分解成3部分-日期、時間和上午/下午
$dt_elements = explode(" " ,$date_time_string);
// 分解日期
$date_elements = explode("/" ,$dt_elements[ 0]);
// 分解時間
$time_elements = explode(":" ,$dt_elements[ 1]);
// 如果是下午,我們將時間增加12小時以便得到24小時制的時間
if ($dt_elements [2]== "PM") { $time_elements[ 0]+=12;}
// 輸出結果
echo mktime ($time_elements [0], $time_elements[ 1], $time_elements[ 2], $date_elements[1], $date_elements[2], $date_elements[0]);

**修改日期

有 時我們需要知道6小時以後是什麼時間,35天前的日期或者從你最後一次玩Quake3後已過去多少秒。我們已經知道如何用mktime()函數從單獨的日 期和時間中獲得Unix的時間戳值。如果我們需要的並非目前日期和時間的Unix時間戳值,我們該咋辦?下面是一些練習可以幫助說明我們後面所要做的。
正如前面所見,mktime()使用以下參數:小時、分、秒、月、天和年。想想第二節,getdate()函數可以為我們獲得這些參數。

// 將目前的時間戳值放入一數組內
$timestamp = time();
echo $timestamp;
echo "p";
$date_time_array = getdate( $timestamp);
// 用mktime()函數重新產生Unix時間戳值
$timestamp = mktime($date_time_array ["hours"], $date_time_array["minutes" ],$date_time_array[ "seconds"],$date_time_array ["mon"], $date_time_array["mday" ],$date_time_array[ "year"]);

echo $timestamp;

看起來有一些令人感到迷惑。我將用一些變量來使上面的程序看起來更容易瞭解。

// 將目前的時間戳值放入一數組內
$timestamp = time();
echo $timestamp;
echo "p";
$date_time_array = getdate( $timestamp);
$hours = $date_time_array[ "hours"];
$minutes = $date_time_array["minutes"];
$seconds = $date_time_array[ "seconds"];
$month = $date_time_array["mon"];
$day = $date_time_array["mday"];
$year = $date_time_array["year"];
// 用mktime()函數重新產生Unix時間戳值
$timestamp = mktime($hours ,$minutes, $seconds,$month ,$day,$year);
echo $timestamp;

現 在我們將由getdate()所產生的時間戳值放入相對應的名稱變量中,所以代碼變得相對容易閱讀和理解。現在如果我們需要在目前的時間上加上19個小 時,我們用$hours+19代替mktime()函數中的$hours。mktime()將自動為我們將時間轉到第二天。

// 將目前的時間戳值放入一數組內
$timestamp = time();
echo strftime( "%Hh%M %A %d %b",$timestamp);
echo "p";
$date_time_array = getdate($timestamp);
$hours = $date_time_array["hours"];
$minutes = $date_time_array["minutes"];
$seconds = $date_time_array["seconds"];
$month = $date_time_array["mon"];
$day = $date_time_array["mday"];
$year = $date_time_array["year"];
// 用mktime()函數重新產生Unix時間戳值
// 增加19小時
$timestamp = mktime($hours + 19, $minutes,$seconds ,$month, $day,$year);
echo strftime( "%Hh%M %A %d %b",$timestamp);
echo "br~E after adding 19 hours";

運行後得到:
14h58 Saturday 03 Jun
09h58 Sunday 04 Jun
~E after adding 19 hours
減少時間也是同樣的-你只需要減少相應變量的值即可。
得到兩個不同時間值的差同樣也是非常簡單。你所需要做的只是將兩個時間值轉換為Unix的時間戳值,然後兩者相減即可。兩者之差即為兩個時間所相隔的秒數。另外一些算法可以很快地將秒轉為天、小時、分和秒。

**為PHP添加DateAdd函數

正如在文章一開始我所說的-寫本文的原因是因為我在PHP中找不到類似ASP的DateDiff函數。在介紹完PHP是如何處理日期和時間,讓我們將ASP中常用的兩個函數移植到PHP。第一個函數是DateAdd。
根據Vbscript的文檔,DateAdd(interval,number,date)函數的定義為「返回已添加指定時間間隔的日期。」
Inetrval為表示要添加的時間間隔字符串表達式,例如分或天;number為表示要添加的時間間隔的個數的數值表達式;Date表示日期。
Interval(時間間隔字符串表達式)可以是以下任意值:
yyyy year年
q Quarter季度
m Month月
y Day of year一年的數
d Day天
w Weekday一週的天數
ww Week of year周
h Hour小時
n Minute分
s Second秒
w、y和d的作用是完全一樣的,即在目前的日期上加一天,q加3個月,ww加7天。

function DateAdd ($interval, $number, $date) {
$date_time_array = getdate($date);
$hours = $date_time_array["hours"];
$minutes = $date_time_array["minutes"];
$seconds = $date_time_array["seconds"];
$month = $date_time_array["mon"];
$day = $date_time_array["mday"];
$year = $date_time_array["year"];
switch ($interval) {
case "yyyy": $year +=$number; break;
case "q": $month +=($number*3); break;
case "m": $month +=$number; break;
case "y":
case "d":
case "w": $day+=$number; break;
case "ww": $day+=($number*7); break;
case "h": $hours+=$number; break;
case "n": $minutes+=$number; break;
case "s": $seconds+=$number; break;
}

$timestamp = mktime($hours ,$minutes, $seconds,$month ,$day, $year);
return $timestamp;}

我們可以將上面的代碼保存為dateadd.inc文件,然後運行以下代碼:

include('dateadd.inc');
$temptime = time();
echo strftime( "%Hh%M %A %d %b",$temptime);
$temptime = DateAdd("n" ,50,$temptime);
echo "p";
echo strftime( "%Hh%M %A %d %b",$temptime);

我們將得到:
15h41 Saturday 03 Jun
16h31 Saturday 03 Jun

為PHP添加DateDiff函數
現在DateAdd已經完成,那麼DateDiff呢?
根據文檔,DateDiff(interval,date1,date2)函數的定義為「返回兩個日期之間的時間間隔」。
Intervals 參數的用法與DateAdd函數中的相同。出於避免過於複雜的考慮,我們決定忽略Vbscript中DateDiff函數中其它複雜的參數,即其兩個可選 的參數變量[firstdayofweek[, firstweekofyear]](它們用於決定星期中第一天是星期天還是星期一和一年中第一週的常數。而且我們只允許intervals有以下五個 值:"w"(周)、"d"(天)、"h"(小時)、"n"(分鐘)和"s"(秒)。

下面的代碼是我們所需要的:

Function DateDiff ($interval, $date1,$date2) {
// 得到兩日期之間間隔的秒數
$timedifference = $date2 - $date1;
switch ($interval) {
case "w": $retval = bcdiv($timedifference ,604800); break;
case "d": $retval = bcdiv( $timedifference,86400); break;
case "h": $retval = bcdiv ($timedifference,3600); break;
case "n": $retval = bcdiv( $timedifference,60); break;
case "s": $retval = $timedifference; break;
}
return $retval;}

將上面的代碼存為datediff.inc文件,然後運行下面的代碼:

include('datediff.inc');
include('dateadd.inc');
$currenttime = time();
echo "Current time: ". strftime("%Hh%M %A %d %b" ,$currenttime)."br";
$newtime = DateAdd ("n",50 ,$currenttime);
echo "Time plus 50 minutes: ". strftime("%Hh%M %A %d %b" ,$newtime)."br";
$temptime = DateDiff ("n",$currenttime ,$newtime);
echo "Interval between two times: ".$temptime;

如果一切順利,你可以看到以下結果:
Current time: 16h23 Saturday 03 Jun
Time plus 50 minutes: 17h13 Saturday 03 Jun
Interval between two times: 50

如 果你在Unix機器上運行PHP,你必須編譯PHP支持BC高精度函數。你必須從以下地址http://www.php.net/extra /number4.tar.gz下載BC庫,然後將其解壓到PHP4的根目錄下,重新編譯PHP,編譯時要加上--enable-bcmath的選 項。(詳細說明見PHP4中README.BCMATH)。PHP4的Windows版本則不需要做任何修補即可直接使用BC高精度函數。
現在你已經得到處理日期和時間的函數,剩下的就是如何將其運用到你的PHP程序中。

【下列文章您可能也有興趣】

沒有留言: