我已经查看过至少二十几个关于此的主题,但还没有真正找到一个好的答案,所以我再次请你再次询问有关重复事件的可怕主题的答案。
我现在每天,每周,每月和每年重复都很好(我仍然需要使用异常事件和诸如此类的东西来修改系统,但它暂时有效)。但是,我们希望能够在每个月,每隔一个月的[第一,第二,第三,第四,第五] [太阳|周一|周二|周三|周四|周五|周六]添加重复活动的能力,每三个月一次。
现在,如果我能理解每个月的逻辑,我可以弄清楚每隔一个月和每三个月。
这是我到目前为止所做的一些事情(注意:我并不是说我有最好的方法可以做到这一点或者其他什么,但系统是我们在不忙于其他人时随着时间的推移而非常缓慢地更新的系统项目,所以我让代码更有效率,因为我有时间)。
首先,我得到格式化日期计算的开始和结束日期:
$ending = $_POST['end_month'] . "/" . $_POST['end_day'] . "/" . substr($_POST['end_year'], 2, 2);
$starting = $_POST['month'] . "/" . $_POST['day'] . "/" . substr($_POST['year'], 2, 2);
然后我得到了这两者之间的区别,知道重复使用一个函数的次数,我相当肯定我在前一段时间找到这个函数并将该数量除以28天以得到它需要多少个TIMES重复,以便每月有一个:
$repeat_number = date_diff($starting, $ending) / 28;
//find the difference in DAYS between the two dates
function date_diff($old_date, $new_date) {
$offset = strtotime($new_date) - strtotime($old_date);
return $offset/60/60/24;
}
然后我将(第一,第二等等)部分添加到[Sun | Mon | etc ...]部分,找出他们想要给我的东西,比如“第一个星期天”:
$find = $_POST['custom_number']. ' ' . $_POST['custom_day'];
然后我使用一个循环,运行需要重复的次数(上面的$ repeat_number):
for($m = 0; $m <= $repeat_number; $m++) {
if($m == 0) {
$month = date('F', substr($starting,0,2));
} else {
$month = date('F', strtotime($month . ' + ' . $m . ' months'));
}
$repeat_date = strtotime($find . ' in ' . $month);
}
现在,我知道这段代码不起作用,我曾经有过一段代码可以找到重复的正确月份和年份,但不一定会找到第一个星期二或者其他任何东西正在寻找。
如果有人能指出我正确的方向,那将是非常感激的。我已经潜入社区一段时间了,刚刚开始尝试积极参与。
提前感谢您提供的任何意见或建议。
答案 0 :(得分:7)
这是您的另一种选择。 strtotime
功能强大,the syntax includes really useful bits类似comprehensive relative time and date wording。
您可以使用格式“of”来生成特定月份的第一个到第N个特定工作日。以下是使用date_create
调用DateTime对象的示例,但常规旧版strtotime
的工作方式相同:
php > $d = date_create('Last Friday of March 2011'); if($d instanceof DateTime) echo $d->format('l F d Y H:i:s');
Friday March 25 2011 00:00:00
php > $d = date_create('First Friday of March 2011'); if($d instanceof DateTime) echo $d->format('l F d Y H:i:s');
Friday March 04 2011 00:00:00
php > $d = date_create('First Sunday of March 2011'); if($d instanceof DateTime) echo $d->format('l F d Y H:i:s');
Sunday March 06 2011 00:00:00
php > $d = date_create('Fourth Sunday of March 2011'); if($d instanceof DateTime) echo $d->format('l F d Y H:i:s');
Sunday March 27 2011 00:00:00
php > $d = date_create('Last Sunday of March 2011'); if($d instanceof DateTime) echo $d->format('l F d Y H:i:s');
Sunday March 27 2011 00:00:00
如果你要求一个无效的月份,它也会溢出:
php > $d = date_create('Ninth Sunday of March 2011'); if($d instanceof DateTime) echo $d->format('l F d Y H:i:s');
Sunday May 01 2011 00:00:00
请注意,它仅适用于ordinal number措辞。不幸的是,你无法通过“第一”或“第三”。
一旦你使用它来获取正常的第N个工作日,你可以根据需要直接添加所需工作日的数量(一周7个,两个14个,三个21个等),直到指定的结束日期或指定的周数已过。再一次,strtotime
救援,使用第二个参数将相对性联系在一起:
php > echo date('l F d Y H:i:s', strtotime('+14 days', strtotime('First Thursday of March 2011')));
Thursday March 17 2011 00:00:00
(我的本地副本也接受'+14 days First Thursday of March 2011'
,但这感觉有点奇怪。)
答案 1 :(得分:3)
我有一个更简单的方法,虽然看起来很糟糕。
我们可以做的是取开始日期,并通过将当天的上限除以7来推断它的“n”值。
例如,如果您的开始日期是18日,我们除以7并取上限来推断它是该月的第3个“任何一天”。
从那里我们可以简单地重复添加7天以获得新的日期,并且每当新日期的天数为7的商数为3时,我们就会达到当月的下一个“第n天”。
下面可以看到一些代码:
$day_nth = ceil(date("j", strtotime($date))/7);
for ($i = 0; $i < $number_of_repeats; $i++) {
while (ceil(date("j", strtotime($date))/7) != $day_nth)
$date = date("Y-m-d", strtotime("+7 days", strtotime($date)));
/* Add your event, or whatever */
/* Now we increment date by 7 days before the loop starts again,
** otherwise the while loop would never execute and we'd just be
** adding to the same date repeatedly. */
$date = date("Y-m-d", strtotime("+7 days", strtotime($date)));
}
这种方法有什么好处,我们甚至不必关心我们正在处理的一周中哪一天。我们只是从开始日期推断所有内容,然后离开。
答案 2 :(得分:0)
我这样做:
//$typeOfDate = 0;
$typeOfDate = 1;
$day = strtotime("2013-02-01");
$to = strtotime(date("Y-m-d", $day) . " +6 month");
if ($typeOfDate == 0) { //use the same day (number) of the month
$dateArray[] = date("Y-m-d", $day);
$event_day_number = date("d", $day);
while ($day <= $to) {
$nextMonth = date("m", get_x_months_to_the_future($day));
$day = strtotime(date("Y-" . $nextMonth . "-d", $day));
$dateArray[] = date("Y-m-d", $day);
}
}
if ($typeOfDate == 0) { //use the same day (like friday) of the month
$dateArray[] = date("Y-m-d", $day);
$monthEvent = date("m", $day);
$day = strtotime(date("Y-m-d", $day) . " +4 weeks");
$newMonthEvent = date("m", $day);
if ($monthEvent == $newMonthEvent) {
$day = strtotime(date("Y-m-d", $day) . " +7 days");
}
while ($day <= $to) {
$dateArray[] = date("Y-m-d", $day);
$monthEvent = date("m", $day);
$day = strtotime(date("Y-m-d", $day) . " +4 weeks");
$newMonthEvent = date("m", $day);
if ($monthEvent == $newMonthEvent) {
$day = strtotime(date("Y-m-d", $day) . " +7 days");
}
}
}