免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 5642 | 回复: 4
打印 上一主题 下一主题

PHP日期处理的经典贴子(转载) [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2004-11-24 12:01 |只看该作者 |倒序浏览
PHP中的日期处理  


作者:Allan Kent  
  
我正打算用PHP编写一种帮助处理系统。我发现我必须知道处理完最后一位客户的问题后已经过去了多长时间?当我过去用ASP时解决这个问题相当简单,ASP有相应的函数DateDiff可以给出两个日期间间隔多少月、多少天和多少秒。当我搜寻完PHP手册后我发现PHP并没有类似的函数。

本文包含以下内容:
1、 得到目前的日期和时间-我们有多少种方式?
2、 改变日期显示的方式-日期和时间的显示形式
3、 转换现在的日期为Unix的时间戳值
4、 改变日期
    a. 增加时间
    b. 减去时间
    c. 找出两日期之间的间隔
5、 为PHP添加DateAdd函数
6、 为PHP添加DateDiff函数

**得到目前的日期和时间

在Unix中,时间的表示方式为计算从1970年1月1日零时起所过去的秒数,这称为UNIX 时间戳(Unix Epoch)。
如果我们有这样一段的代码:
  1. <?
  2. echo time();
  3. ?>;
复制代码

将返回值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代码为:
  1. <?
  2. echo date ("Hhi l d F" ,time());
  3. ?>;
复制代码

当我们执行这段代码,我们发现我们所得到的结果为:
180609 Sunday 21 May
这样的结果看起来有些奇怪。让我们再查一下PHP手册,原来’h’所代表的是12 小时制的小时数。这再次证明了一句真理:“计算机只做你所告诉它该做的,而不是你想要它做的”。我们有两个选择。第一个是在h前使用转义字符“\”:
  1. echo date ("H\hi l d F", time());
复制代码

我们得到这样的结果:

18h12 Sunday 21 May

这正是我们所要的。但如果我们在一个十分复杂的句子中需要包含日期和时间,我们是否需要对每个字符使用转义字符?
答案当然是不。我们使用另一个函数strftime()。
strftime()有两个好处。
第一个好处我们并不在本文讨论范围内-如果你使用setlocale()函数,你可以通过strftime得到相应语言的月份的名称。
另外的一个好处是你可以将特别的日期和时间的格式化字符包含在你的字符串中。这同时也意味着无论你是否要学习date()函数的所有特殊格式化字符,你都必须学习一整套完全不同的格式化字符。

strftime()工作的方式和date()没有什么不同,除了特殊格式化字符的前面必须添加一个百分号%。如果用strftime()函数,前面例子的代码如下:
  1. <?
  2. echo strftime ("%Hh%M %A %d %b" ,time());
  3. ?>;
复制代码


结果为:
18h24 Sunday 21 May

这也许看起来将简化繁,但考虑一下如果你所需要的显示的为
"Today is Sunday 21 May 2000. The time is somewhere close to 18h24."
我想使用date()函数无疑令人感到厌烦。

在开始的时候,我提及我们有两种方式可以从Unix时间戳值中得到有用的数据。我们刚刚了解了date()和strftime()。另一个getdate()。这个函数只需要Unix 的时间戳值作为参数,而函数的返回值为日期和时间的数组。
下面是一个例子:
  1. <?
  2. $date_time_array = getdate (time());
  3. echo $date_time_array[ "weekday"];
  4. ?>;
复制代码


返回的结果为:
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] );
从左往右你必须提供小时、分、秒、月、天和年。最后一个参数用于指定你是否处于夏令时,此参数是可选的,所以我们将忽略它。
代码如下:
  1. <?
  2. echo mktime (0, 0,0 ,5, 27,2000 );
  3. ?>;
复制代码

由于不知道小时、分和秒同时这些参数必须填写,我将其设置为0。设置为0意味着时间为午夜。
  1. <?
  2. $access_date = "2000/05/27";
  3. //explode()函数用一个字符串作为分界来分解另一个字符串。这个例子$access_date通过字符串”/”来分解
  4. $date_elements = explode("/" ,$access_date);
  5. // 此时
  6. // $date_elements[0] = 2000
  7. // $date_elements[1] = 5
  8. // $date_elements[2] = 27
  9. echo mktime (0, 0,0 ,$date_elements [1], $date_elements[ 2],$date_elements [0]);
  10. ?>;
复制代码

我们看一个比从Access数据库单纯获得日期更复杂的情况,我们得到一个以下格式的日期和时间:2000/05/27 02:40:21 PM

  1. <?
  2. // 来自Access的字符串
  3. $date_time_string = "2000/05/27 02:40:21 PM";
  4. // 将字符串分解成3部分-日期、时间和上午/下午
  5. $dt_elements = explode(" " ,$date_time_string);
  6. // 分解日期
  7. $date_elements = explode("/" ,$dt_elements[ 0]);
  8. // 分解时间
  9. $time_elements = explode(":" ,$dt_elements[ 1]);
  10. // 如果是下午,我们将时间增加12小时以便得到24小时制的时间
  11. if ($dt_elements [2]== "PM") { $time_elements[ 0]+=12;}
  12. // 输出结果
  13. echo mktime ($time_elements [0], $time_elements[ 1], $time_elements[ 2], $date_elements[1], $date_elements[2], $date_elements[0]);
  14. ?>;
复制代码


**修改日期

有时我们需要知道6小时以后是什么时间,35天前的日期或者从你最后一次玩Quake3后已过去多少秒。我们已经知道如何用mktime()函数从单独的日期和时间中获得Unix的时间戳值。如果我们需要的并非目前日期和时间的Unix时间戳值,我们该咋办?下面是一些练习可以帮助说明我们后面所要做的。
正如前面所见,mktime()使用以下参数:小时、分、秒、月、天和年。想想第二节,getdate()函数可以为我们获得这些参数。

  1. <?
  2. // 将目前的时间戳值放入一数组内
  3. $timestamp = time();
  4. echo $timestamp;
  5. echo "p";
  6. $date_time_array = getdate( $timestamp);
  7. // 用mktime()函数重新产生Unix时间戳值
  8. $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"]);
  9. echo $timestamp;
  10. ?>;
复制代码


看起来有一些令人感到迷惑。我将用一些变量来使上面的程序看起来更容易了解。

  1. <?
  2. // 将目前的时间戳值放入一数组内
  3. $timestamp = time();
  4. echo $timestamp;
  5. echo "p";
  6. $date_time_array = getdate( $timestamp);
  7. $hours = $date_time_array[ "hours"];
  8. $minutes = $date_time_array["minutes"];
  9. $seconds = $date_time_array[ "seconds"];
  10. $month = $date_time_array["mon"];
  11. $day = $date_time_array["mday"];
  12. $year = $date_time_array["year"];
  13. // 用mktime()函数重新产生Unix时间戳值
  14. $timestamp = mktime($hours ,$minutes, $seconds,$month ,$day,$year);
  15. echo $timestamp;
  16. ?>;
复制代码


现在我们将由getdate()所产生的时间戳值放入相对应的名称变量中,所以代码变得相对容易阅读和理解。现在如果我们需要在目前的时间上加上19个小时,我们用$hours+19代替mktime()函数中的$hours。mktime()将自动为我们将时间转到第二天。

  1. <?
  2. // 将目前的时间戳值放入一数组内
  3. $timestamp = time();
  4. echo strftime( "%Hh%M %A %d %b",$timestamp);
  5. echo "p";
  6. $date_time_array = getdate($timestamp);
  7. $hours = $date_time_array["hours"];
  8. $minutes = $date_time_array["minutes"];
  9. $seconds = $date_time_array["seconds"];
  10. $month = $date_time_array["mon"];
  11. $day = $date_time_array["mday"];
  12. $year = $date_time_array["year"];
  13. // 用mktime()函数重新产生Unix时间戳值
  14. // 增加19小时
  15. $timestamp = mktime($hours + 19, $minutes,$seconds ,$month, $day,$year);
  16. echo strftime( "%Hh%M %A %d %b",$timestamp);
  17. echo "br~E after adding 19 hours";
  18. ?>;
复制代码


运行后得到:
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天。

  1. <?
  2. function DateAdd ($interval, $number, $date) {
  3. $date_time_array = getdate($date);
  4. $hours = $date_time_array["hours"];
  5. $minutes = $date_time_array["minutes"];
  6. $seconds = $date_time_array["seconds"];
  7. $month = $date_time_array["mon"];
  8. $day = $date_time_array["mday"];
  9. $year = $date_time_array["year"];
  10. switch ($interval) {
  11. case "yyyy": $year +=$number; break;
  12. case "q": $month +=($number*3); break;
  13. case "m": $month +=$number; break;
  14. case "y":
  15. case "d":
  16. case "w": $day+=$number; break;
  17. case "ww": $day+=($number*7); break;
  18. case "h": $hours+=$number; break;
  19. case "n": $minutes+=$number; break;
  20. case "s": $seconds+=$number; break;
  21. }
  22. $timestamp = mktime($hours ,$minutes, $seconds,$month ,$day, $year);
  23. return $timestamp;}
  24. ?>;
复制代码


我们可以将上面的代码保存为dateadd.inc文件,然后运行以下代码:

  1. <?
  2. include('dateadd.inc');
  3. $temptime = time();
  4. echo strftime( "%Hh%M %A %d %b",$temptime);
  5. $temptime = DateAdd("n" ,50,$temptime);
  6. echo "p";
  7. echo strftime( "%Hh%M %A %d %b",$temptime);
  8. ?>;
复制代码


我们将得到:
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"(秒)。

Let's see what we can come up with: 下面的代码是我们所需要的:

  1. <?
  2. Function DateDiff ($interval, $date1,$date2) {
  3. // 得到两日期之间间隔的秒数
  4. $timedifference = $date2 - $date1;
  5. switch ($interval) {
  6. case "w": $retval = bcdiv($timedifference ,604800); break;
  7. case "d": $retval = bcdiv( $timedifference,86400); break;
  8. case "h": $retval = bcdiv ($timedifference,3600); break;
  9. case "n": $retval = bcdiv( $timedifference,60); break;
  10. case "s": $retval = $timedifference; break;
  11. }
  12. return $retval;}
  13. ?>;
复制代码


将上面的代码存为datediff.inc文件,然后运行下面的代码:
  1. <?
  2. include('datediff.inc');
  3. include('dateadd.inc');
  4. $currenttime = time();
  5. echo "Current time: ". strftime("%Hh%M %A %d %b" ,$currenttime)."br";
  6. $newtime = DateAdd ("n",50 ,$currenttime);
  7. echo "Time plus 50 minutes: ". strftime("%Hh%M %A %d %b" ,$newtime)."br";
  8. $temptime = DateDiff ("n",$currenttime ,$newtime);
  9. echo "Interval between two times: ".$temptime;
  10. ?>;
复制代码


如果一切顺利,你可以看到以下结果:
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程序中。

论坛徽章:
0
2 [报告]
发表于 2004-11-24 14:27 |只看该作者

PHP日期处理的经典贴子(转载)

mysql 里的 timestamp 数据 和 php  time() 生成的时间戳,怎么转换?

论坛徽章:
0
3 [报告]
发表于 2004-11-24 16:29 |只看该作者

PHP日期处理的经典贴子(转载)

timestamp to time:
<?php
$time=strtotime("$timestamp";
?>;

time to timestamp:

<?php
$timestamp=date("Y-m-d H:i:s",$time);
?>;
二楼的是这样吗?

论坛徽章:
0
4 [报告]
发表于 2004-11-24 16:44 |只看该作者

PHP日期处理的经典贴子(转载)

不是
我的意思是把 yyyymmddhhmmss 各式的数据
转成: 从 1970 年 1 月 1 日 算起的秒数

论坛徽章:
0
5 [报告]
发表于 2004-11-24 17:58 |只看该作者

PHP日期处理的经典贴子(转载)

顺帖一个比较有用的函数:

-----------------------------------------------------------
<?
function dur($big, $small)
{
        $diff          = $big - $small;
        $daysDiff = floor($diff / 60 / 60 / 24);
       
        $diff         -= $daysDiff * 60 * 60 * 24;
        $hrsDiff  = floor($diff / 60 / 60);
       
        $diff    -= $hrsDiff * 60 * 60;
        $minsDiff = floor($diff / 60);
       
        $diff    -= $minsDiff * 60;
        $secsDiff = $diff;

        return  $daysDiff."Day, ".$hrsDiff."Hour, ".$minsDiff."Sec";
}

echo dur(mktime(0,0,0, 11, 23, 2004), mktime(0,0,0,11,22,2004));
?>;
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP