暢談Perl時(shí)間處理函數(shù)用法
本文和大家重點(diǎn)討論一下Perl時(shí)間處理函數(shù)的概念,Perl能在絕大多數(shù)操作系統(tǒng)運(yùn)行,可以方便地向不同操作系統(tǒng)遷移,并且Perl借取了C、sed、awk、shellscripting以及很多其他程序語(yǔ)言的特性。
Perl時(shí)間處理函數(shù)
表示日期的方式多種多樣:
“18Jan1973″,”18/01/1973″,”01/18/1973″,”Jan181973″,”18-01-73″,”18-01-1973″,”01/73″,其中一些格式意思不清(如”01-06-1973″是表示6月1日呢,還是表示1月6日呢?)如果不規(guī)定日期的表示形式,是很難處理的。
想了解”18Jan1973″和”6Sep1950″之間的區(qū)別,須要把它們轉(zhuǎn)換為數(shù)字表示。Unix內(nèi)部運(yùn)用紀(jì)元秒表示時(shí)間。日期和時(shí)間加起來(lái)表示之自格林威志時(shí)間1970年1月1日午夜時(shí)分(紀(jì)元)到當(dāng)前時(shí)刻之間的秒數(shù)。”18Jan1973″(假定為午夜時(shí)分)的紀(jì)元秒為96163200。在該系統(tǒng)中,午夜表示一天的開(kāi)始時(shí)刻。
讓我們生成一個(gè)日期通過(guò)Perl中提供的gmtime函數(shù),你可以自己來(lái)驗(yàn)證這點(diǎn)。給定一個(gè)用以表示自從紀(jì)元以來(lái)的秒數(shù)的整數(shù),通過(guò)gmtime函數(shù)可以計(jì)算出代表相應(yīng)的日期和時(shí)刻,例如:
Perl-le‘printscalargmtime96163200′
ThuJan1800:00:001973
調(diào)用gmtime()函數(shù),你會(huì)得到一系列值的列表,包括時(shí),分,秒,日期,月份,年份等等。
Perl-le‘printjoin(”,”,gmtime96163200)’
0,0,0,18,0,73,4,17,0
前面3個(gè)0分別表示秒,分,時(shí)。小時(shí)是從0-23,故下午是12時(shí)往后。第4個(gè)數(shù)表示該月中的天數(shù)(本例中為18號(hào))。第5個(gè)數(shù)表示月份,從0開(kāi)始(代表1月份)。之所以從0開(kāi)始,是因?yàn)樵路輰?duì)應(yīng)著月份數(shù)組的下標(biāo):
@months=qw(JanFebMarAprMayJunJulAugSepOctNovDec);$month=(gmtime96163200)[4];#“Jan”
年份(本例中為73)的表示有點(diǎn)特殊。它并不是年份的***兩位數(shù)字。它表示從1900年開(kāi)始的年份。為什么要這樣表示呢?這是因?yàn)镃語(yǔ)言就是這樣處理的。Perl試圖使得其庫(kù)和系統(tǒng)調(diào)用盡量接近操作系統(tǒng)的處理方式。
所以,如果你想輸出4位數(shù)的年份,表示如下:
$year=(gmtime96163200)[5]+1900;
如果你不了解這種處理方式,就會(huì)制造出Y2K疑問(wèn),你也許會(huì)這樣寫(xiě):
$year=“19″.(gmtime96163200)[5];
#出錯(cuò)!2000年將變?yōu)?9100
對(duì)于gmtime()函數(shù)的返回值還沒(méi)有介紹完,還有4,17,和0這3個(gè)數(shù)。它們分別表示一星期中的第幾天(星期日為0),一年中的第幾天(0表示一年中的***天),以及能不能采用夏時(shí)制(表示不采用,正數(shù)表示采用,負(fù)數(shù)表示不可知)。
Perl中的time()函數(shù)返回以紀(jì)元秒形式表示的當(dāng)前日期和時(shí)間。如果你打算把它轉(zhuǎn)換為字符串,就可運(yùn)用gmtime()和localtime()函數(shù):$now=localtime(time());
($sec,$min,$hour,$day,$mon,$year,$wday,$yday,$isdst)=localtime(time());
如果調(diào)用localtime()或gmtime()時(shí)不帶參數(shù),它將自己調(diào)用time()
$now=localtime();
($sec,$min,$hour,$day,$mon,$year,$wday,$yday,$isdst)=localtime();
Perl時(shí)間處理函數(shù)中多見(jiàn)的日期和時(shí)間操作
如果你打算計(jì)算兩個(gè)時(shí)刻之間的時(shí)間段,只需將它們轉(zhuǎn)換為相應(yīng)的紀(jì)元秒,然后兩數(shù)相減即可:
$difference_in_seconds=$later_datetime-$earlier_datetime;
要把秒轉(zhuǎn)換為分,時(shí),或天數(shù),只須要分別將它們除以60,3600和86400即可:
$difference_in_minutes=$difference_in_seconds/60;
$difference_in_hours=$difference_in_seconds/3600;
$difference_in_day=$difference_in_seconds/86400;
反過(guò)來(lái)做,你也可以回答如下疑問(wèn):”4天后是幾號(hào)?”:
$then=time()+86400*4;
printscalarlocaltime$then;
它給出的答案精確到秒。例如,如果4天后的紀(jì)元秒值為932836935,你可以輸出日期的字符串如下;
SatJul2411:23:171999
如果你打算輸出那個(gè)日期的午夜時(shí)分(如”SatJul2400:00:001999″)運(yùn)用如下模塊:$then=$then-$then%86400;#去掉那個(gè)日期的尾巴
類(lèi)似地,你可以用四舍五入法,輸出最靠近午夜時(shí)分的日期:
$then+=43200;#addonhalfaday
$then=$then-$then%86400;#truncatetotheday
如果你的時(shí)區(qū)距離GMT為相差偶數(shù)個(gè)小時(shí),這就管用了。并不是所有的時(shí)區(qū)都是很容易處理的。你所真實(shí)須要的是在你自己的時(shí)區(qū)內(nèi)計(jì)算紀(jì)元秒,而不是在GMT中計(jì)算。
Perl中的名為T(mén)ime::Local的模塊,可以提供兩個(gè)函數(shù)timelocal()和timegm()。其返回值同localtime()和gmtime()一樣。
useTime::Local;
$then=time()+4*86400;
$then=timegmlocaltime$then;
#localepochseconds$then-=$then%86400;
#truncatetotheday
$then=timelocalgmtime$then;
#backtogmtepochseconds
printscalarlocaltime$then,“\n”。#p#
Perl時(shí)間處理函數(shù)中日常生活所用的日期和時(shí)間的表示
你已經(jīng)級(jí)掌握了時(shí),分,年等值的意思,也了解了紀(jì)元秒的意思。而日常生活中的日期和時(shí)間是用字符串來(lái)表示的,你怎樣才能把日常所用的日期和時(shí)間串格式轉(zhuǎn)換成紀(jì)元秒呢?
要領(lǐng)之一是寫(xiě)出語(yǔ)法分析小程序,該要領(lǐng)靈活而高速:
useTime::Local;
@months{qw(JanFebMarAprMayJun
JulAugSepOctNovDec)}=(0..11);
$_=“19Dec199715:30:02″;
/(\d\d)\s+(\w+)\s+(\d+)\s+(\d+):(\d+):(\d+)/
ordie“Notadate”;
$mday=$1;
$mon=exists($months{$2})?$months{$2}:die“Badmonth”;
$year=$3-1900;
($h,$m,$s)=($4,$5,$6);
$epoch_seconds=timelocal($s,$m,$h,$mday,$mon,$year);
一個(gè)更通用些的要領(lǐng),是從CPAN安裝Date::Manip模塊。
useDate::Manip;
$epoch_seconds=UnixDate(”19Dec199715:30:02″,”s”);
留心,由于Date::Manip是個(gè)大模塊,運(yùn)用該模塊時(shí),將會(huì)添加你的程序的啟動(dòng)時(shí)間。其中一個(gè)原由是Date::Manip將對(duì)多種不同的格式執(zhí)行識(shí)別,如:
“today”
“now”
“firstsundayinapril2000″
“3:15,today”
“3:15pm,firstsundayinapril2000″
“2000/01/1809:15″DateManipulation
2036,2037,2038,…,1901?!
大多數(shù)C程序把紀(jì)元秒存為有符號(hào)整數(shù),可表示正的和負(fù)的日期,但計(jì)算機(jī)存儲(chǔ)器所表示的整數(shù)大小是有限的,用有限的位數(shù)來(lái)表示秒。這就是說(shuō),我們?cè)谟?jì)算紀(jì)元秒時(shí),所表示的日期是有限定的。
確切的限度取決于你的機(jī)器所能表示的整數(shù)的位數(shù)。Perl最多以32位的長(zhǎng)度存儲(chǔ)整數(shù)。粗略地講,有一位用來(lái)表示正負(fù)號(hào),其余31位來(lái)表示數(shù)。如果8位,你可以存儲(chǔ)的***數(shù)是255,即2的8次方減1。故Perl中所存儲(chǔ)的32位符號(hào)數(shù)中的***數(shù)為:
print2**31-1,“\n”;
2147483647
這個(gè)數(shù)字對(duì)應(yīng)了哪個(gè)日期呢?
printscalar(gmtime2**31-1),“\n”;
TueJan1903:14:072038
在那個(gè)時(shí)刻的1秒之后會(huì)發(fā)生什么呢?
printscalar(gmtime2**31),“\n”;
FriDec1320:45:521901
對(duì)于32位有符號(hào)整數(shù)來(lái)說(shuō),2**31太大了。它”翻卷過(guò)去了”,其符號(hào)位被置為負(fù)號(hào),因而成為了所能表示的***負(fù)數(shù)。這對(duì)應(yīng)于1970年開(kāi)始時(shí)刻之前的秒的***值。
其結(jié)果說(shuō)明了什么呢?你不能存儲(chǔ)gmtime(2**31)之前或gmtime(2**31-1)之后的以紀(jì)元秒表示的日期。
你可千萬(wàn)不要想不開(kāi),這可不是什么大疑問(wèn)。如果你要用到32位有符號(hào)整數(shù)表示的紀(jì)元秒以外的時(shí)間,你只須要改動(dòng)你的表示方式,你可從CPAN中找到不少日期模塊,其中的Date::Calc和Date::Manip很可能是功能***的兩個(gè)模塊。
這兩個(gè)模塊運(yùn)用自己的日期表示方式,以防止Y1901-Y2038的限定。Date::Manip運(yùn)用羅馬歷法,從公元0000到公元9999。Date::Calc也運(yùn)用羅馬歷法,可表示的年份從1到32767。
總結(jié)
Perl時(shí)間處理函數(shù)中對(duì)于在1902-2037范圍內(nèi)的日期和時(shí)期表示,把它們轉(zhuǎn)換為紀(jì)元秒,要存取這些數(shù),你只需運(yùn)用整數(shù)算術(shù)運(yùn)算,gmtime()和localtime()函數(shù),以及標(biāo)準(zhǔn)的Time::Local模塊。如果要對(duì)該范圍以外的日期執(zhí)行計(jì)算或者要分析某特殊的日期格式,你可以運(yùn)用CPAN中的Date::Manip和Date::Calc模塊。
【編輯推薦】
- 用Perl POE實(shí)現(xiàn)端口重定向
- 淺談配置Eclipse支持Perl腳本開(kāi)發(fā)
- 用Perl和Google Earth創(chuàng)建可用性地圖
- 在 Perl/Tk 中使用高級(jí)窗口小部件
- 讓Perl成為你的嵌入式開(kāi)發(fā)工具