我有一个包含n
个字符串的数组,格式为YYYY-MM-DD(例如,“2010-10-31”)。
如何将日期与此数组中的字符串进行比较?
例如,删除30天前的字符串?
我有一个包含n
个字符串的数组,格式为YYYY-MM-DD(例如,“2010-10-31”)。
如何将日期与此数组中的字符串进行比较?
例如,删除30天前的字符串?
YYYY-MM-DD
格式的日期非常好比较,只需要用简单的字符串比较就可以了。在Perl中,使用lt
和gt
操作符进行比较。YYYY-MM-DD
字符串转换为“正确”的日期对象、时期时间等的开销。#!/usr/bin/env perl
use strict;
use warnings;
my $thirty_days = 30 * 24 * 60 * 60;
my ($old_day, $old_month, $old_year) = (localtime(time - $thirty_days))[3..5];
my $cutoff = sprintf('%04d-%02d-%02d',
$old_year + 1900, $old_month + 1, $old_day);
my @dates = ('2010-10-12', '2010-09-12', '2010-08-12', '2010-09-13');
for my $date (@dates) {
print "$date\n" if $date gt $cutoff;
}
2010-02-29
,虽然在字典上比较但仍是错误的日期。一个日期包会指出源日期错误,这似乎应该是脚本的一部分,不是吗? - dawg$date gt $cutoff
将失败,原因如下:1)在sprintf
中使用除“-”以外的分隔符;2)字符串前面有空格,但应该匹配;3)字符串后面有空格,但不应该匹配。几乎所有的CPAN日期解析库都可以处理这三种情况,以及许多其他情况。一旦检查了分隔符并去掉了前导和尾随空格,你真的节省了那么多“开销”或代码吗?Date::Calc是用C编写的,非常快。Date::Time功能非常丰富。两者都检查有效的日期输入,而这种方法则不会。 - dawg猜测有很多方法可以做到这一点,但我喜欢Date::Simple来处理这种事情。
以下是文档中的一个例子:
use Date::Simple ('date', 'today');
# Difference in days between two dates:
$diff = date('2001-08-27') - date('1977-10-05');
# Offset $n days from now:
$date = today() + $n;
print "$date\n"; # uses ISO 8601 format (YYYY-MM-DD)
它非常适用于对对象++进行算术运算。
只限日期,不包括小时、分钟或秒数。
use strict; use warnings;
use DateTime ();
use DateTime::Duration ();
use DateTime::Format::Natural ();
my $parser = DateTime::Format::Natural->new;
my $now = DateTime->now;
my $delta = DateTime::Duration->new( days => 30 );
my $cutoff = $now->subtract_duration( $delta );
my @new_dates = map { $_->[1] }
grep { -1 == $_->[0] }
map {
chomp;
[
DateTime->compare(
$parser->parse_datetime( $_ ),
$cutoff
),
$_
]
} <DATA>;
print "@new_dates";
__DATA__
2010-07-31
2010-08-31
2010-09-30
2010-10-31
my $now = DateTime->now;
,无需解析 localtime
的输出。 - daotoadif ($year < $out2[$j][7]) {
$lt = 1;
goto JUMP;
}
if ($year > $out2[$j][7]) {
$gt = 1;
goto JUMP;
}
if ($month < $out2[$j][5]) {
$lt = 1;
goto JUMP;
}
if ($month > $out2[$j][5]) {
$gt = 1;
goto JUMP;
}
if ($day < $out2[$j][6]) {
$lt = 1;
goto JUMP;
}
if ($day > $out2[$j][6]) {
$gt = 1;
goto JUMP;
}
if ($time < $out2[$j][4]) {
$lt = 1;
goto JUMP;
}
if ($time > $out2[$j][4]) {
$gt = 1;
goto JUMP;
}
JUMP:
if ($lt == 1) {
$out2[$j][2] = "$dtime $month\/$day\/$year";
$out2[$j][4] = $time;
$out2[$j][5] = $month;
$out2[$j][6] = $day;
$out2[$j][7] = $year;
$lt = 0;
}
if ($gt == 1) {
$out2[$j][3] = "$dtime $month\/$day\/$year";
$out2[$j][4] = $time;
$out2[$j][5] = $month;
$out2[$j][6] = $day;
$out2[$j][7] = $year;
$gt = 0;
}
一个好的开始是阅读Perl的许多日期和DateTime网站。
YYYY-MM-DD格式是ISO 8601日期表示的一种形式。有些变体被认为是可接受的,例如YYYY-MM-DD
和YYYYMMDD
甚至是旧数据中的YYMM
。在选择比较这些日期的方法之前,您应该查看权威列表。
如果ISO 8601日期字符串:1)是有效日期;2)具有相同的格式,带或不带-
分隔符;3)缺少前导和尾随空格,则一个吸引人的特性是您可以使用简单的字典字符串比较对字符串进行排序或比较。
总的来说:
--- 否则 ---
决定使用CPAN模块解析日期字符串(或自己匹配),
如果您的日期在该范围内,请转换为时期时间(或使用Date::Manip或Date::Calc等执行更大规模的日期/时间操作的CPAN模块)
对于时间类型(时期时间、绝对天数等),执行算术运算
将时间转换回所需的格式...
以下是执行此操作的代码:
use warnings; use strict;
use Date::Calc qw/:all/;
my (@date_strings, @abs_days);
my $target=Date_to_Days(2010, 1, 15);
# set @date_string to "YYYY-MM-DAY" between some dates
for my $AbsDay(Date_to_Days(2009,1,1)..Date_to_Days(2011,12,31)) {
my ($year, $mon, $day)=Add_Delta_Days(1,1,1,$AbsDay-1);
my $s="$year-$mon-$day";
push @date_strings, $s;
}
foreach my $s (@date_strings) {
my ($year, $mon, $day);
if(($year, $mon, $day)=$s=~/^(\d+)-(\d+)-(\d+)/) {
my $days=Date_to_Days($year, $mon, $day);
push @abs_days, $days
if ($target-$days <= 30 && $target-$days >= -30 );
}
}
print "absolute day way:\n";
foreach my $days (@abs_days) {
my ($year, $mon, $day)=Add_Delta_Days(1,1,1,$days-1);
print "$year-$mon-$day\n";
}
您可以使用Time::ParseDate模块,
use strict;
use warning;
use Time::ParseDate;
my @dates = ('2010-10-12', '2010-09-14', '2010-08-12', '2010-09-13');
my @dates =
grep {parsedate($_, NO_RELATIVE => 1, UK => 1) > parsedate('-30 days') }@dates;
#output: 2010-10-12 2010-09-14
为什么不使用CORE自5.10以来的Time::Piece和Time::Seconds,而不是CPAN搜索结果中的前几个?
use strict;
use warnings;
use Time::Piece (); # we don't need to include overloaded locatime
use Time::Seconds;
use Data::Dumper;
my @dates = qw/2010-10-31 2012-10-16 2011-09-08/;
my $now = Time::Piece::localtime();
my @date_objects = map {
Time::Piece->strptime( $_, '%F') # %F is the same as %Y-%m-%d
} @dates;
my @filtered_dates = grep {
$now - $_ < (ONE_DAY * 30)
} @date_objects;
print Dumper(map($_->strftime('%F'), @filtered_dates));
var minDate = ...;
var maxDate = ...;
foreach my $date ( @$dates ) {
if ($minDate gt $date){ # Less than.
$minDate = $date; # Minimal date.
}
if ($minDate lt $date){ # Greater than.
$minDate = $date; # Maxamal date.
}
}