一. Perl時(shí)間的表示函數(shù)
1. 表示日期的方式多種多樣:
"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ù)。
如, "18 Jan 1973:(假定為午夜時(shí)分)的紀(jì)元秒為96163200。
2. 在該系統(tǒng)中,午夜表示一天的開始時(shí)刻。
讓我們通過(guò)Perl中提供的gmtime函數(shù)生成一個(gè)日期。
給定一個(gè)用以表示自從紀(jì)元以來(lái)的秒數(shù)的整數(shù), 通過(guò)gmtime函數(shù)可以計(jì)算出代表相應(yīng)的日期和時(shí)刻,
例一:
調(diào)用gmtime()函數(shù),你會(huì)得到一系列值的列表,包括時(shí),分,秒,日期,月份,年份等.
#!/usr/bin/perl
use Time::localtime;
$t_num = 96163200;
$tm = scalar(gmtime($t_num));
print $tm,"\n";
輸出:
Thu Jan 18 00:00:00 1973
例二:以","為分隔符輸出時(shí)間
print join(",", gmtime(96163200));
0,0,0,18,0,73,4,17,0
語(yǔ)義:
前3個(gè)數(shù): 0,0,0, 分別表示秒, 分, 時(shí). 小時(shí)是從0-23,故下午是12時(shí)往后.
第4個(gè)數(shù): 18, 表示該月中的天數(shù)(本例中為18號(hào))。
第5個(gè)數(shù): 0 , 表示月份,從0開始(代表1月份)。
之所以從0開始,是因?yàn)樵路輰?duì)應(yīng)著月份數(shù)組的下標(biāo):
@months = qw(Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec);
$month = @months[(gmtime($t_num))[4]];
print "MONTH: ",$month,"\n";
第6個(gè)數(shù): 73, 年份, (本例中為73)的表示有點(diǎn)特殊。它并不是年份的最后兩位數(shù)字。
它表示從1900年開始的年份。
為什么要這樣表示呢?
這是因?yàn)镃語(yǔ)言就是這樣處理的。
Perl試圖使得其庫(kù)和系統(tǒng)調(diào)用盡量接近操作系統(tǒng)的處理方式。
所以,如果你想輸出4位數(shù)的年份,表示如下:
$year=(gmtime(96163200))[5]+1900;
如果你不了解這種處理方式,就會(huì)制造出Y2K疑問(wèn),你也許會(huì)這樣寫:
$year="19".(gmtime(96163200))[5]; #出錯(cuò)!2000年將變?yōu)?9100
第7個(gè)數(shù): 4 , 表示一星期中的第幾天(星期日為0).
第8個(gè)數(shù): 17, 一年中的第幾天(0表示一年中的第一天).
第9個(gè)數(shù): 0 , 能不能采用夏時(shí)制(0表示不采用,正數(shù)表示采用,負(fù)數(shù)表示不可知).
3. 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ù)中(日期和時(shí)間操作)
1. 計(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;
2. 計(jì)算"4天后是幾號(hào)?":
$then=time()+86400*4;
print scalar(localtime($then));
它給出的答案精確到秒。
例如,
如果4天后的紀(jì)元秒值為932836935,你可以輸出日期的字符串如下;
Sat Jul 24 11:23:17 1999
3. 輸出某個(gè)日期的午夜時(shí)分
如"Sat Jul 24 00:00:00 1999",
運(yùn)用如下模塊:
$then=$then-$then%86400;#去掉那個(gè)日期的尾巴
類似地,你可以用四舍五入法,輸出最靠近午夜時(shí)分的日期:
$then += 43200; #add on half a day
$then = $then - $then%86400; #truncate to the day
如果你的時(shí)區(qū)距離GMT為相差偶數(shù)個(gè)小時(shí),這就管用了。
并不是所有的時(shí)區(qū)都是很容易處理的。
你所真實(shí)須要的是在你自己的時(shí)區(qū)內(nèi)計(jì)算紀(jì)元秒,而不是在GMT中計(jì)算。
Perl中的名為Time::Local的模塊,
可以提供兩個(gè)函數(shù)timelocal()和timegm()。其返回值同localtime()和gmtime()一樣。
use Time::Local;
$then = time() + 4*86400;
$then = timegm(localtime($then)); #local epoch seconds
$then -= $then%86400; #truncate to the day
$then = timelocal(gmtime($then)); #back to gmt epoch seconds
print scalar(localtime$then,“\n”。
三. Perl時(shí)間處理函數(shù)中日常生活所用的日期和時(shí)間的表示
前面介紹了時(shí),分,年等值的意思,也了解了紀(jì)元秒的意思。
而日常生活中的日期和時(shí)間是用字符串來(lái)表示的,
怎樣才能把日常所用的日期和時(shí)間串格式轉(zhuǎn)換成紀(jì)元秒呢?
1. 要領(lǐng)之一是寫出語(yǔ)法分析小程序,該要領(lǐng)靈活而高速:
#!/usr/bin/perl
use Time::Local;
@months{qw(Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec)}=(0..11);
$_ = "19 Dec 1997 15:30:02";
/(\d\d)\s+(\w+)\s+(\d+)\s+(\d+):(\d+):(\d+)/ or die "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);
print "day: ",$mday,"\n";
print "mon: ",$mon,"\n";
print "year: ",$year,"\n";
print "seconds: ",$epoch_seconds,"\n";
2. 一個(gè)更通用些的要領(lǐng),是從CPAN安裝Date::Manip模塊。
useDate::Manip;
$epoch_seconds=UnixDate("19 Dec 1997 15:30:02","s");
留心,由于Date::Manip是個(gè)大模塊,運(yùn)用該模塊時(shí),將會(huì)添加你的程序的啟動(dòng)時(shí)間。
其中一個(gè)原由是Date::Manip將對(duì)多種不同的格式執(zhí)行識(shí)別,
如:
"today"
"now"
"first sunday in april 2000"
"3:15,today"
"3:15 pm,first sunday in april 2000"
"2000/01/18 09:15" Date Manipulation
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ù)為:
print 2**31-1,"\n";
2147483647
這個(gè)數(shù)字對(duì)應(yīng)了哪個(gè)日期呢?
print scalar(gmtime(2**31-1)),"\n";
Tue Jan 19 03:14:07 2038
在那個(gè)時(shí)刻的1秒之后會(huì)發(fā)生什么呢?
print scalar(gmtime(2**31)),"\n";
Fri Dec 13 20:45:52 1901
對(duì)于32位有符號(hào)整數(shù)來(lái)說(shuō),2**31太大了。
它"翻卷過(guò)去了",其符號(hào)位被置為負(fù)號(hào),因而成為了所能表示的最大負(fù)數(shù)。
這對(duì)應(yīng)于1970年開始時(shí)刻之前的秒的最大值。
其結(jié)果說(shuō)明了什么呢?你不能存儲(chǔ)gmtime(2**31)之前或gmtime(2**31-1)之后的以紀(jì)元秒表示的日期。
你可千萬(wàn)不要想不開,這可不是什么大疑問(wèn)。
如果你要用到32位有符號(hào)整數(shù)表示的紀(jì)元秒以外的時(shí)間,你只須要改動(dòng)你的表示方式,
你可從CPAN中找到不少日期模塊,其中的Date::Calc和Date::Manip很可能是功能最強(qiáng)的兩個(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模塊。
您可能感興趣的文章:- Perl5和Perl6對(duì)比使用Sigils的差別
- Perl6中的垃圾收集
- 強(qiáng)大的Perl正則表達(dá)式實(shí)例詳解
- Perl中的符號(hào) ->;、=>; 和 :: 分別表示什么意思?
- Perl中常見符號(hào)與操作
- Perl學(xué)習(xí)教程之單行命令詳解
- Perl字符串處理函數(shù)大全
- 詳解linux下批量替換文件內(nèi)容的三種方法(perl,sed,shell)
- Perl與JS的對(duì)比分析(數(shù)組、哈希)
- 使用Perl生成隨機(jī)密碼
- ASP.NET中HyperLink超鏈接控件的使用方法
- 將Perl5代碼遷移到Perl6上的解決方案