在每个月的第 n 个工作日重复事件
Posted
技术标签:
【中文标题】在每个月的第 n 个工作日重复事件【英文标题】:Repeating Events on the "nth" Weekday of Every Month 【发布时间】:2011-07-19 19:41:39 【问题描述】:我已经查看了至少 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);
现在,我知道此代码不起作用,我曾有一次代码可以为重复找到正确的月份和年份,但不一定会找到第一个星期二或其他任何内容正在寻找。
如果有人能指出我正确的方向,将不胜感激。我已经在社区潜伏了一段时间,最近才开始尝试积极参与。
提前感谢您提供的任何意见或建议。
【问题讨论】:
不要忘记从月底开始支持相同的模式,或者至少每个月最后一个。 你应该重命名你的date_diff
函数——当你升级到php 5.3时,它会和the built in date_diff
function冲突。
感谢查尔斯提供的信息。我不知道他们添加了 date_diff 函数。
【参考方案1】:
这里有一个可能的替代方案。 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 措辞。很遗憾,你不能通过“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'
,但感觉有点奇怪。)
【讨论】:
提醒其他打算使用它的人,strtotime 的“of”部分不适用于 PHP v4.3.9。 @ericissocial,你为什么还要编写为那个史前 PHP 版本设计的代码? PHP4 已经死了多年,并且不受支持。 我们的 CMS 和客户端在旧服务器上,因为在过去 5 年左右的时间里,我的老板一直在那里拥有一切。我实际上已经说服他我们需要移动它,只需做一些准备就可以将控制所有客户站点的系统移动到新服务器(并且需要一些工作来让人们为改变做好准备)。不过,您的回答在我的测试服务器和我们计划将 CMS 移至的服务器上产生了奇迹。我的手指交叉,在下周我将不必再处理 PHP 4.3.9。 请注意,其中大部分仅在 5.3 之后可用。如果您在 2013 年 3 月的最后一个星期一尝试过,它将无法在 5.2 中运行【参考方案2】:我有一个更简单的方法,虽然它可能看起来很老套。
我们可以做的是获取开始日期,并通过将当天的上限除以 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)));
这种方法的好处在于,我们甚至不必关心我们正在处理一周中的哪一天。我们只是从开始日期推断一切,然后离开。
【讨论】:
谢谢,我愚蠢地使用了floor($start / 7) + 1
,这在 7 日、14 日、21 日和 28 日是错误的,但在其他日子还可以 :)【参考方案3】:
我是这样写的:
//$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");
【讨论】:
以上是关于在每个月的第 n 个工作日重复事件的主要内容,如果未能解决你的问题,请参考以下文章
使用当月 VBA 的日期填充行中的一系列单元格,然后在下个月的第一天保存数据并清除工作表
每月第 5 个工作日的 TimeTrigger - 计划的 Web 作业 Azure