У меня есть 2 даты в PHP, как я могу запустить цикл foreach, чтобы пройти все эти дни?

207

Я начинаю с даты 2010-05-01и заканчиваю 2010-05-10. Как я могу перебрать все эти даты в PHP?

Shamoon
источник

Ответы:

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-е, установите $end11-е. Вы можете настроить формат по своему вкусу. Смотрите руководство по PHP для DatePeriod .

Гордон
источник
2
хорошая новость - есть патч для установки флага, включающего дату окончания, которая (скрестив пальцы) сделает его в будущей версии.
Salathe
8
$begin->setTime(0,0); $end->setTime(12,0);или инициализация со временем дня начальной даты, так как любое время позже, чем время конечной даты, будет включать конечную дату в цикл. Не самое стильное решение, но это лучший вариант, если нет правильного флага.
Крис
31
Если вы хотите включить дату окончания в ваш интервал, вы можете сделать: $ end = $ end-> modify ('+1 день');
JulienITARD
3
Можно ли использовать это, но повернуть вспять, чтобы вернуться в историю?
Джон
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. Но вам нужны дальнейшие модификации для их использования.
Хенк Поли
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, чтобы защитить от пропуска или повторения дней.

Harold1983-
источник
4
Мне не нравится внешний вид этого 86400. Я понимаю, что это 60 * 60 * 24, но все же ... что-то в этом раздражает меня.
MikeD
13
в этом случае это работает, но если есть переключение между обычным и экономящим время солнечным светом, оно потерпит неудачу, потому что есть 90000 второго дня, которые у вас будут дважды в цикле ...
oezi
2
Майк, лучше всего настроить константу и назвать ее «ДЕНЬ», чтобы ее стало намного легче читать.
Pixel Developer
5
Это будет страдать от проблем летнего времени. Когда вы пересекаете точку перехода на летнее время, она облажается. 12:00 не 12:00 по обе стороны от момента времени.
Эрик Коуп
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% медленнее за 60 итераций). Но я выбираю этот для ретро-совместимости для старых платформ хостинга.
Ifnot
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
user2182143
источник
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". Оба будут работать.

zukayu
источник
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 \ Carbon;
  • использовать 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)));
    }

?>
Сотерис 92
источник