我在 PHP 中有 2 个日期,我怎样才能运行一个 foreach 循环来度过所有这些日子?

Posted

技术标签:

【中文标题】我在 PHP 中有 2 个日期,我怎样才能运行一个 foreach 循环来度过所有这些日子?【英文标题】:I have 2 dates in PHP, how can I run a foreach loop to go through all of those days? 【发布时间】:2011-03-13 13:45:28 【问题描述】:

我从日期2010-05-01 开始,以2010-05-10 结束。如何在 php 中遍历所有这些日期?

【问题讨论】:

【参考方案1】:

转换为 unix 时间戳可以更轻松地在 php 中进行日期数学运算:

$startTime = strtotime( '2010-05-01 12:00' );
$endTime = strtotime( '2010-05-10 12:00' );

// Loop between timestamps, 24 hours at a time
for ( $i = $startTime; $i <= $endTime; $i = $i + 86400 ) 
  $thisDate = date( 'Y-m-d', $i ); // 2010-05-01, 2010-05-02, etc

在使用具有 DST 的时区的 PHP 时,请确保添加一个不是 23:00、00:00 或 1:00 的时间,以防止日期跳过或重复。

【讨论】:

我不喜欢 86400 的外观。我知道它是 60 * 60 * 24,但仍然......它的某些地方让我感到厌烦。 在这种情况下,它可以工作,但是如果在正常时间和阳光节约时间之间切换,它将失败,因为您的循环中有两次 90000 次日...... 迈克,最好的办法是设置一个常量并将其命名为“DAY”,这样它就更容易阅读了。 这将受到夏令时问题的影响。当您越过夏令时时间点时,它将被搞砸。 12:00am 不是时间点两边的 12:00am。 这段代码(每个代码每天 86400 秒)确实有夏令时问题!夏令时有些日子只持续 23 小时,有些则持续 25 小时。【参考方案2】:
$startTime = strtotime('2010-05-01'); 
$endTime = strtotime('2010-05-10'); 

// Loop between timestamps, 1 day at a time 
$i = 1;
do 
   $newTime = strtotime('+'.$i++.' days',$startTime); 
   echo $newTime;
 while ($newTime < $endTime);

$startTime = strtotime('2010-05-01'); 
$endTime = strtotime('2010-05-10'); 

// Loop between timestamps, 1 day at a time 
do 
   $startTime = strtotime('+1 day',$startTime); 
   echo $startTime;
 while ($startTime < $endTime);

【讨论】:

看来这个解决方案比接受的答案慢(未运行一些工作台:60 次迭代慢 100%)。但我选择这个是为了与旧的托管平台兼容。【参考方案3】:
$begin = new DateTime('2010-05-01');
$end = new DateTime('2010-05-10');

$interval = DateInterval::createFromDateString('1 day');
$period = new DatePeriod($begin, $interval, $end);

foreach ($period as $dt) 
    echo $dt->format("l Y-m-d H:i:s\n");

这将输出$start$end 之间定义的时间段内的所有日期。如果要包括第 10 个,请将$end 设置为第 11 个。您可以根据自己的喜好调整格式。请参阅DatePeriod 的 PHP 手册。它需要 PHP 5.3。

【讨论】:

好消息 - 有一个补丁用于设置一个标志以包含结束日期(手指交叉)将使其成为未来版本。 $begin-&gt;setTime(0,0); $end-&gt;setTime(12,0); 或使用开始日期的时间进行初始化,因为任何晚于结束日期的时间都将在循环中包含结束日期。不是最时尚的修复,但只要没有合适的标志,它就是最佳选择。 如果你想在间隔中包含结束日期,你可以这样做:$end = $end->modify('+1 day'); 是否可以使用这个但反过来,循环回历史? @JulienITARD 这是一个不错的主意,但更优雅的是 $end->add( $interval ) 因为它直接响应更改的间隔;)【参考方案4】:

使用此功能:-

function dateRange($first, $last, $step = '+1 day', $format = 'Y-m-d' ) 
                $dates = array();
                $current = strtotime($first);
                $last = strtotime($last);

                while( $current <= $last )     
                    $dates[] = date($format, $current);
                    $current = strtotime($step, $current);
                
                return $dates;
        

用法/函数调用:-

增加一天:-

dateRange($start, $end); //increment is set to 1 day.

按月增加:-

dateRange($start, $end, "+1 month");//increase by one month

如果您想设置日期格式,请使用第三个参数:-

   dateRange($start, $end, "+1 month", "Y-m-d H:i:s");//increase by one month and format is mysql datetime

【讨论】:

【参考方案5】:

这也包括最后日期

$begin = new DateTime( "2015-07-03" );
$end   = new DateTime( "2015-07-09" );

for($i = $begin; $i <= $end; $i->modify('+1 day'))
    echo $i->format("Y-m-d");

如果您不需要最后日期,只需从条件中删除 =

【讨论】:

一定要注意$begin在循环之后会不同。此循环修改由new DateTime( "2015-07-03" ) 创建的对象。因此,为什么您应该使用 DateTimeImmutable 版本。但是您需要进一步修改才能使用它们。【参考方案6】:

有一个办法:

 $date = new Carbon();
 $dtStart = $date->startOfMonth();
 $dtEnd = $dtStart->copy()->endOfMonth();

 $weekendsInMoth = [];
 while ($dtStart->diffInDays($dtEnd)) 

     if($dtStart->isWeekend()) 
            $weekendsInMoth[] = $dtStart->copy();
     

     $dtStart->addDay();
 

$weekendsInMoth 的结果是周末的数组!

【讨论】:

【参考方案7】:

inclusive 范围的 php.net 示例中复制:

$begin = new DateTime( '2012-08-01' );
$end = new DateTime( '2012-08-31' );
$end = $end->modify( '+1 day' ); 

$interval = new DateInterval('P1D');
$daterange = new DatePeriod($begin, $interval ,$end);

foreach($daterange as $date)
    echo $date->format("Ymd") . "<br>";

【讨论】:

这是最好最完整的答案。仅缺少对 DateInterval 值 P1D 的一些解释,因此这里有一些周期指示符示例 两天:P2D 两秒:PT2S 一周十分钟:P1WT10M Y 表示年份 M 表示月份 D 表示几天 W 表示几周。这些被转换成天,所以不能与 D.H 组合小时 M 分钟 S 秒【参考方案8】:

这是另一个简单的实现 -

/**
 * Date range
 *
 * @param $first
 * @param $last
 * @param string $step
 * @param string $format
 * @return array
 */
function dateRange( $first, $last, $step = '+1 day', $format = 'Y-m-d' ) 
    $dates = [];
    $current = strtotime( $first );
    $last = strtotime( $last );

    while( $current <= $last ) 

        $dates[] = date( $format, $current );
        $current = strtotime( $step, $current );
    

    return $dates;

例子:

print_r( dateRange( '2010-07-26', '2010-08-05') );

Array (
    [0] => 2010-07-26
    [1] => 2010-07-27
    [2] => 2010-07-28
    [3] => 2010-07-29
    [4] => 2010-07-30
    [5] => 2010-07-31
    [6] => 2010-08-01
    [7] => 2010-08-02
    [8] => 2010-08-03
    [9] => 2010-08-04
    [10] => 2010-08-05
)

【讨论】:

亲爱的哈迪,谢谢伙计 :)【参考方案9】:
$date = new DateTime($_POST['date']);
$endDate = date_add(new DateTime($_POST['date']),date_interval_create_from_date_string("7 days"));

while ($date <= $endDate) 
    print date_format($date,'d-m-Y')." AND END DATE IS : ".date_format($endDate,'d-m-Y')."\n";
    date_add($date,date_interval_create_from_date_string("1 days"));

您也可以像这样进行迭代,$_POST['date'] 可以从您的应用程序或网站中删除 除了$_POST['date'],你也可以把你的字符串放在"21-12-2019"。两者都可以。

【讨论】:

【参考方案10】:

如果您使用 Laravel 并想使用 Carbon,则正确的解决方案如下:

$start_date = Carbon::createFromFormat('Y-m-d', '2020-01-01');
$end_date = Carbon::createFromFormat('Y-m-d', '2020-01-31');

$period = new CarbonPeriod($start_date, '1 day', $end_date);

foreach ($period as $dt) 
 echo $dt->format("l Y-m-d H:i:s\n");

记得添加:

使用碳\碳; 使用 Carbon\CarbonPeriod;

【讨论】:

【参考方案11】:
<?php

    $start_date = '2015-01-01';
    $end_date = '2015-06-30';

    while (strtotime($start_date) <= strtotime($end_date)) 
        echo "$start_daten";
        $start_date = date ("Y-m-d", strtotime("+1 days", strtotime($start_date)));
    

?>

【讨论】:

【参考方案12】:

对于Carbon 用户

use Carbon\Carbon;

$startDay = Carbon::parse("2021-08-01");
$endDay= Carbon::parse("2021-08-05");
$period = $startDay->range($endDay, 1, 'day');

当我打印数据时

[
     Carbon\Carbon @1627790400 #4970
       date: 2021-08-01 00:00:00.0 America/Toronto (-04:00),
     ,
     Carbon\Carbon @1627876800 #4974
       date: 2021-08-02 00:00:00.0 America/Toronto (-04:00),
     ,
     Carbon\Carbon @1627963200 #4978
       date: 2021-08-03 00:00:00.0 America/Toronto (-04:00),
     ,
     Carbon\Carbon @1628049600 #5007
       date: 2021-08-04 00:00:00.0 America/Toronto (-04:00),
     ,
     Carbon\Carbon @1628136000 #5009
       date: 2021-08-05 00:00:00.0 America/Toronto (-04:00),
     ,
]

这是使用 dd($period-&gt;toArray()); 的 Laravel 数据转储。如果需要,您现在可以使用 foreach 语句遍历 $period

一个重要说明 - 它包括提供给方法的两个边缘日期。

有关更酷的日期相关内容,请查看Carbon docs。

【讨论】:

这是最简单的方法

以上是关于我在 PHP 中有 2 个日期,我怎样才能运行一个 foreach 循环来度过所有这些日子?的主要内容,如果未能解决你的问题,请参考以下文章

具有一个日期和三个不同值的数据框:我怎样才能得到中间的那个?

VB.NET - 给定一个日期,我怎样才能得到最后四个星期五的日期?

将php变量传递给角度

怎样在本地运行PHP

我怎样才能在未来的某个特定时间运行一个函数? [关闭]

php停止已经运行的脚本进程