一. Perl時間的表示函數(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)部運用紀(jì)元秒表示時間。
日期和時間加起來表示:
自格林威志時間1970年1月1日午夜時分(紀(jì)元)到當(dāng)前時刻之間的秒數(shù)。
如, "18 Jan 1973:(假定為午夜時分)的紀(jì)元秒為96163200。
2. 在該系統(tǒng)中,午夜表示一天的開始時刻。
讓我們通過Perl中提供的gmtime函數(shù)生成一個日期。
給定一個用以表示自從紀(jì)元以來的秒數(shù)的整數(shù), 通過gmtime函數(shù)可以計算出代表相應(yīng)的日期和時刻,
例一:
調(diào)用gmtime()函數(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
例二:以","為分隔符輸出時間
print join(",", gmtime(96163200));
0,0,0,18,0,73,4,17,0
語義:
前3個數(shù): 0,0,0, 分別表示秒, 分, 時. 小時是從0-23,故下午是12時往后.
第4個數(shù): 18, 表示該月中的天數(shù)(本例中為18號)。
第5個數(shù): 0 , 表示月份,從0開始(代表1月份)。
之所以從0開始,是因為月份對應(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個數(shù): 73, 年份, (本例中為73)的表示有點特殊。它并不是年份的最后兩位數(shù)字。
它表示從1900年開始的年份。
為什么要這樣表示呢?
這是因為C語言就是這樣處理的。
Perl試圖使得其庫和系統(tǒng)調(diào)用盡量接近操作系統(tǒng)的處理方式。
所以,如果你想輸出4位數(shù)的年份,表示如下:
$year=(gmtime(96163200))[5]+1900;
如果你不了解這種處理方式,就會制造出Y2K疑問,你也許會這樣寫:
$year="19".(gmtime(96163200))[5]; #出錯!2000年將變?yōu)?9100
第7個數(shù): 4 , 表示一星期中的第幾天(星期日為0).
第8個數(shù): 17, 一年中的第幾天(0表示一年中的第一天).
第9個數(shù): 0 , 能不能采用夏時制(0表示不采用,正數(shù)表示采用,負數(shù)表示不可知).
3. Perl中的time()函數(shù)返回以紀(jì)元秒形式表示的當(dāng)前日期和時間。
如果你打算把它轉(zhuǎ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ù)中(日期和時間操作)
1. 計算兩個時刻之間的時間段,
只需將它們轉(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;
2. 計算"4天后是幾號?":
$then=time()+86400*4;
print scalar(localtime($then));
它給出的答案精確到秒。
例如,
如果4天后的紀(jì)元秒值為932836935,你可以輸出日期的字符串如下;
Sat Jul 24 11:23:17 1999
3. 輸出某個日期的午夜時分
如"Sat Jul 24 00:00:00 1999",
運用如下模塊:
$then=$then-$then%86400;#去掉那個日期的尾巴
類似地,你可以用四舍五入法,輸出最靠近午夜時分的日期:
$then += 43200; #add on half a day
$then = $then - $then%86400; #truncate to the day
如果你的時區(qū)距離GMT為相差偶數(shù)個小時,這就管用了。
并不是所有的時區(qū)都是很容易處理的。
你所真實須要的是在你自己的時區(qū)內(nèi)計算紀(jì)元秒,而不是在GMT中計算。
Perl中的名為Time::Local的模塊,
可以提供兩個函數(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ù)中日常生活所用的日期和時間的表示
前面介紹了時,分,年等值的意思,也了解了紀(jì)元秒的意思。
而日常生活中的日期和時間是用字符串來表示的,
怎樣才能把日常所用的日期和時間串格式轉(zhuǎn)換成紀(jì)元秒呢?
1. 要領(lǐng)之一是寫出語法分析小程序,該要領(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. 一個更通用些的要領(lǐng),是從CPAN安裝Date::Manip模塊。
useDate::Manip;
$epoch_seconds=UnixDate("19 Dec 1997 15:30:02","s");
留心,由于Date::Manip是個大模塊,運用該模塊時,將會添加你的程序的啟動時間。
其中一個原由是Date::Manip將對多種不同的格式執(zhí)行識別,
如:
"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ì)元秒存為有符號整數(shù),可表示正的和負的日期;
但計算機存儲器所表示的整數(shù)大小是有限的, 用有限的位數(shù)來表示秒.
這就是說,我們在計算紀(jì)元秒時, 所表示的日期是有限定的。
確切的限度取決于你的機器所能表示的整數(shù)的位數(shù)。
Perl最多以32位的長度存儲整數(shù)。
粗略地講,有一位用來表示正負號,其余31位來表示數(shù)。
如果8位,你可以存儲的最大數(shù)是255,即2的8次方減1。
故Perl中所存儲的32位符號數(shù)中的最大數(shù)為:
print 2**31-1,"\n";
2147483647
這個數(shù)字對應(yīng)了哪個日期呢?
print scalar(gmtime(2**31-1)),"\n";
Tue Jan 19 03:14:07 2038
在那個時刻的1秒之后會發(fā)生什么呢?
print scalar(gmtime(2**31)),"\n";
Fri Dec 13 20:45:52 1901
對于32位有符號整數(shù)來說,2**31太大了。
它"翻卷過去了",其符號位被置為負號,因而成為了所能表示的最大負數(shù)。
這對應(yīng)于1970年開始時刻之前的秒的最大值。
其結(jié)果說明了什么呢?你不能存儲gmtime(2**31)之前或gmtime(2**31-1)之后的以紀(jì)元秒表示的日期。
你可千萬不要想不開,這可不是什么大疑問。
如果你要用到32位有符號整數(shù)表示的紀(jì)元秒以外的時間,你只須要改動你的表示方式,
你可從CPAN中找到不少日期模塊,其中的Date::Calc和Date::Manip很可能是功能最強的兩個模塊。
這兩個模塊運用自己的日期表示方式,以防止Y1901-Y2038的限定。
Date::Manip運用羅馬歷法,從公元0000到公元9999。
Date::Calc也運用羅馬歷法,可表示的年份從1到32767。
總結(jié)
Perl時間處理函數(shù)中對于在1902-2037范圍內(nèi)的日期和時期表示,把它們轉(zhuǎn)換為紀(jì)元秒,
要存取這些數(shù),你只需運用整數(shù)算術(shù)運算,gmtime()和localtime()函數(shù),以及標(biāo)準(zhǔn)的Time::Local模塊。
如果要對該范圍以外的日期執(zhí)行計算或者要分析某特殊的日期格式,
你可以運用CPAN中的Date::Manip和Date::Calc模塊。
您可能感興趣的文章:- Perl5和Perl6對比使用Sigils的差別
- Perl6中的垃圾收集
- 強大的Perl正則表達式實例詳解
- Perl中的符號 ->;、=>; 和 :: 分別表示什么意思?
- Perl中常見符號與操作
- Perl學(xué)習(xí)教程之單行命令詳解
- Perl字符串處理函數(shù)大全
- 詳解linux下批量替換文件內(nèi)容的三種方法(perl,sed,shell)
- Perl與JS的對比分析(數(shù)組、哈希)
- 使用Perl生成隨機密碼
- ASP.NET中HyperLink超鏈接控件的使用方法
- 將Perl5代碼遷移到Perl6上的解決方案