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