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