この記事では、Perl の時間処理関数の概念に焦点を当てます。Perl は、ほとんどのオペレーティング システムで実行でき、さまざまなオペレーティング システムに簡単に移行できます。また、Perl は、C、sed、awk、シェルスクリプト、その他多くのプログラミング言語の機能を借用しています。
1. Perl の時刻表現関数
1. 日付を表す方法はたくさんあります:
"18Jan1973";
"18/01/1973";
"01/18/1973";
"Jan181973" ;
"18-01-73";
"18-01-1973";
"01/73"。
一部の形式は不明瞭です (6 月 1 日を意味する「01-06-1973」など)。
日付表現が指定されていないと対応が難しくなります
「1973年1月18日」と「1950年9月6日」の違いを理解したい場合は、数値表現に変換する必要があります.
Unix エポック秒は時間を表すために内部的に使用されます。
日付と時刻の合計は以下を表します:
グリニッジ標準時 1970 年 1 月 1 日の午前 0 時 (エポック) と現在の瞬間の間の秒数。
例: 「1973 年 1 月 18 日: エポック秒 (午前 0 時であると想定) は 96163200 です。
2。このシステムでは、午前 0 時は 1 日の始まりを表します。
提供されている gmtime 関数を通じて生成しましょう。 Perl では日付。エポックからの秒数を表す整数が与えられると、対応する日付と時刻は gmtime 関数を使用して計算できます。 例 1:
gmtime() 関数を呼び出すと、一連の結果が得られます。時、分、秒、日付、月、年などを含む値のリスト
#!/usr/bin/perl use Time::localtime; $t_num = 96163200; $tm = scalar(gmtime($t_num)); print $tm,"\n";
例 2: 出力の区切り文字として「,」を使用します。時間
0,0,0,18,0,73,4,17,0print 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;
セマンティクス:
4 番目の数字 18 は、その月の日数を表します。 5 番目の数字: 0 は、0 から始まる月を表します (1 月を表します)。
0 から始まる理由は、月が month 配列の添字に対応するためです:
$now=localtime(time()); ($sec,$min,$hour,$day,$mon,$year,$wday,$yday,$isdst)=localtime(time());
$year=(gmtime(96163200))[ 5]+1900;
$ year="19".(gmtime(96163200))[5] ; #エラー! 2000 年は 19100 になります
7 番目の数値: 4、曜日を表します (日曜日は 0)。
9 番目の数字: 0、夏時間を採用できるかどうか (0 は夏時間を使用しないことを意味し、正の数は夏時間を使用することを意味し、負の数は夏時間を採用することを意味します)
3. Perl の time () 関数は、現在の日付と時間をエポック秒で返します
文字列に変換する場合は、gmtime() と localtime() を使用できます。関数:
$now=localtime();
($sec,$min,$hour,$day,$mon,$year,$wday,$yday,$isdst)=localtime();
$difference_in_minutes=$difference_in_seconds/60; $difference_in_hours=$difference_in_seconds/3600; $difference_in_day=$difference_in_seconds/86400;
2. Perl の時刻処理関数 (日付と時刻の操作)
1. 2 つの瞬間の期間を計算します。
それらを対応するエポック秒に変換し、2 つの数値を減算します。 、時間、または日、それぞれを 60、3600、および 86400 で割る必要があります:
$then=time()+86400*4; print scalar(localtime($then));
$then += 43200; #add on half a day $then = $then - $then%86400; #truncate to the day
Sat Jul 24 11:23:17 1999
3. 特定の日付の午前 0 時を出力します。 「Sat Jul 24 00:00:00 1999」のように、
次のモジュールを使用します:
同様に、四捨五入メソッドを使用して次のこともできます。真夜中に最も近い日付を出力します。 日付:
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”。
これは、タイムゾーンが GMT から偶数時間離れている場合に機能します。
すべてのタイムゾーンに対応するのが簡単なわけではありません。
Perl の Time::Local という名前のモジュール は、timelocal() と timegm() という 2 つの関数を提供できます。戻り値は localtime() および gmtime() と同じです。
🎜🎜🎜#!/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=; $mon=exists($months{})?$months{}:die"Badmonth"; $year=-1900; ($h,$m,$s)=(,,); $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";
前面介绍了时,分,年等值的意思,也了解了纪元秒的意思。
而日常生活中的日期和时间是用字符串来表示的,
怎样才能把日常所用的日期和时间串格式转换成纪元秒呢?
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模块。
以上がPerlの時間処理関数の使い方を詳しく解説の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。