If you have the seconds-since-epoch integer value, like
time produces, it's easy to get the date of any number of seconds ago. Problems can arise due to Daylight Saving Time, but starting out from (around) noon on any day, you can always reliably derive the date for $x days in the past. Depending on whether you started out using timegm() or timelocal(), you can use
gmtime() or
localtime() to regenerate a string — possibly complemented with
POSIX' strftime(), for more formatting options.
Starting from a string, there are various date parsing modules available. Personally, I like Time::Local, a standard module, to convert date parts back into seconds since epoch. A bit of regexp manipulation, and you can easily pull the date parts out of the string, and you can complement them with your own values, such as 12 for the hour, for solving that DST problem I mentioned above.
for ($year, $month, $day) {
if (/\b\d{1}\b/) {
$_ = "0$_";
}
}
Don't do that, use
sprintf instead.
sprintf "%02d", $n formats any number with 2 digits, using a leading 0 if necessary. And you can use it to produce a whole date string consisting of multiple parts, at once. Or like I said, use strftime() from
POSIX.
As an example, here's my code that does the equivalent from yours, for dates between 1970 and 2035 (or so). Much shorter!
#!/usr/local/bin/perl -wl
use Time::Local;
use POSIX 'strftime';
sub get_yesterday {
my $date = shift;
my ($year, $month, $day) = split/-/,$date;
my $time = timegm(0, 0, 12, $day, $month-1, $year);
return strftime "%y-%m-%d", gmtime($time - 24*60*60);
}
print get_yesterday("2005-04-01");
I get:
05-03-31