我在PHP中有2个日期,如何运行foreach循环来遍历所有这些日子?


207

我以日期开始,以2010-05-01结束2010-05-10。如何在PHP中迭代所有这些日期?

Answers:


535

需要PHP5.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个。您可以根据自己的喜好调整格式。参见PHP Manual for DatePeriod


2
好消息-有一个修补程序可以设置一个标记,以包括结束日期(手指越过),该日期将使其成为将来的版本。
萨拉

8
$begin->setTime(0,0); $end->setTime(12,0);或以开始日期的当天时间晚于结束日期的时间进行初始化,则会在循环中包括结束日期。不是最时尚的修复程序,但是只要没有适当的标记,它就是最佳选择。
克里斯(Chris)

31
如果要将结束日期包括在间隔中,则可以执行以下操作:$ end = $ end-> modify('+1 day');
JulienITARD 2014年

3
是否可以使用此功能但将其反转以返回历史记录?
乔恩(Jon)

3
@JulienITARD多数民众赞成在一个不错的主意,但更优雅的将是$ end-> add($ interval),因为它直接响应更改的时间间隔;)
GDY

92

这还包括最后日期

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

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


1
请务必注意,循环后$begin会有所不同。此循环修改由创建的对象new DateTime( "2015-07-03" )。因此,为什么要使用DateTimeImmutable版本。但是您需要对其进行一些进一步的修改。
Henk Poley19年

40

转换为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
}

将PHP与具有DST的时区一起使用时,请确保添加的时间不是23:00、00:00或1:00,以防止跳过或重复几天。


4
我不喜欢那种86400的外观。我知道它是60 * 60 * 24,但是...关于它的某些东西让我感到不舒服。
MikeD 2010年

13
在这种情况下,它可以工作,但是如果在正常时间和节省日光时间之间进行切换,则将失败,因为第二天有90000次您将在循环中两次...
oezi 2010年

2
迈克,最好的办法是设置一个常量并将其命名为“ DAY”,这样它就更容易阅读了。
Pixel开发人员

5
这将遇到夏令时问题。当您越过夏令时时,就会搞砸了。时间点的两边不是12:00 am。
埃里克·科普

1
此代码确实(每个代码每天有86400秒)在夏时制方面有问题!使用夏令时,某些日子只能持续23个小时,而有些则可持续25个小时。
sbrbot

20

从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,成秒
Orcra

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

2
看来,此解决方案比公认的答案要慢(未进行测试):60次迭代要慢100%。但是我选择此版本是为了与旧托管平台具有复古兼容性。
如果不是2013年

7

这是另一个简单的-

/**
 * 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
)

5

用户此功能:

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

2

这是一种方法:

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

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

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

     $dtStart->addDay();
 }

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


0
$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"。两者都会起作用。


0

如果您使用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;

0
<?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)));
    }

?>
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.