У мене є два дати в PHP, як я можу запустити цикл foreach, щоб пройти всі ці дні?


207

Я починаю з побачення 2010-05-01і закінчую 2010-05-10. Як я можу повторити всі ці дати в PHP?

Відповіді:


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
хороша новина - є патч для встановлення прапора, щоб включити дату закінчення, яка (пальці схрещені) перетворить його в майбутню версію.
салат

8
$begin->setTime(0,0); $end->setTime(12,0);або ініціалізація з часом дня дати початку, оскільки будь-який час пізніший за дату закінчення буде включати дату закінчення в циклі. Не найстильніший виправлення, але це найкращий варіант, якщо немає правильного прапора.
Кріс

31
Якщо ви хочете включити дату закінчення до свого інтервалу, ви можете зробити: $ end = $ end-> модифікувати ("+1 день");
JulienITARD

3
Чи можна використовувати це, але відмінити його, щоб повернути назад в історію?
Джон

3
@JulienITARD - це дуже гарна ідея, але більш елегантною буде $ end-> add ($ інтервал), оскільки вона відповідає безпосередньо на змінений інтервал;)
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, щоб захистити від пропускання чи повторення днів.


4
Мені не подобається зовнішній вигляд цього 86400. Я розумію, що це 60 * 60 * 24, але все ж ... щось із цього мене дратує.
MikeD

13
у цьому випадку вона працює, але якщо є перемикання між нормальним та сонячним світлом, це вийде з ладу, тому що є 90000 секунд, які у вас будуть двічі в циклі ...
oezi

2
Майку, найкраще - це налаштувати константу і назвати її "ДЕНЬ", щоб читати стало набагато простіше.
The 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, місяців M для днів W для тижнів W. Вони перетворюються на дні, тому їх не можна поєднувати з D. H протягом годин M, хвилин S протягом секунд
Оркра

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
Здається, що це рішення повільніше, ніж прийнята відповідь (запустив кілька стендів: на 100% повільніше за 60 ітерацій). Але я вибираю цю для ретро-сумісності для старих платформ для розміщення хостингу.
Якщо ні

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

?>
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.