7

我已经查看了至少 2 打关于此的主题,但还没有真正找到一个好的答案,所以我来找你再次询问有关重复事件的可怕主题的答案。

我现在每天、每周、每月和每年的重复工作都很好(我仍然需要用异常事件和诸如此类的东西来改进系统,但它暂时可以工作)。但是,我们希望能够添加在每个月的(1、2、3、4、5)[Sun|Mon|Tue|Wed|Thu|Fri|Sat] 每隔一个月重复事件的能力,并且每三个月。

现在,如果我能理解每个月的逻辑,我就能算出每隔一个月和每三个月。

这是我到目前为止所拥有的一些东西(注意:我并不是说我有最好的方法或任何东西,但是当我们不忙于其他项目时,我们会随着时间的推移非常缓慢地更新系统,所以我有时间让代码更有效率)。

首先,我将开始和结束日期格式化为日期计算:

$ending = $_POST['end_month'] . "/" . $_POST['end_day'] . "/" . substr($_POST['end_year'], 2, 2);
$starting = $_POST['month'] . "/" . $_POST['day'] . "/" . substr($_POST['year'], 2, 2);

然后我得到这两者之间的差异,以知道使用我相当确定我一段时间前在这里找到的函数重复多少次,并将该数量除以 28 天以获得大约需要重复多少次这样一个月有一次:

$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;
}

然后我将 (1st,2nd,etc...) 部分添加到 [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);
}

现在,我知道这段代码不起作用,我曾经有一次代码可以为重复出现正确的月份和年份,但不一定会找到第一个星期二或正在寻找的任何内容.

如果有人能指出我正确的方向,将不胜感激。我已经潜伏在社区有一段时间了,最​​近才开始尝试积极参与。

提前感谢您提供的任何意见或建议。

4

3 回答 3

8

这是您可能的替代方案。 strtotime功能强大,并且语法包括非常有用的位,例如全面的相对时间和日期措辞

您可以使用它使用“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

请注意,它仅适用于序数词。不幸的是,你不能通过“1st”或“3rd”。

一旦您使用它来获取正确的第 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',但感觉有点奇怪。)

于 2011-03-21T17:02:40.590 回答
3

我有一个更简单的方法,虽然它可能看起来很hack-ish。

我们可以做的是获取开始日期,并通过将当天的上限除以 7 来推断它的“n”值。

例如,如果您的开始日期是 18 日,我们除以 7 并取上限来推断它是该月的第 3 个“任意一天”。

从那里我们可以简单地重复添加 7 天以获得新日期,并且每当新日期的上限 7 商为 3 时,我们就达到了该月的下一个“第 nthwhatever-day”。

一些代码如下所示:

$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)));
}

这种方法的好处是我们甚至不必关心我们正在处理一周中的哪一天。我们只是从开始日期推断一切,然后离开。

于 2014-01-15T19:08:27.710 回答
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");
                    }
                }
            }
于 2013-02-26T22:06:12.117 回答