Mam 2 daty w PHP, jak mogę uruchomić pętlę Foreach, aby przejść przez te wszystkie dni?

207

Zaczynam od daty, 2010-05-01a kończę na 2010-05-10. Jak mogę iterować wszystkie te daty w PHP?

Shamoon
źródło

Odpowiedzi:

535

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

Spowoduje to wygenerowanie wszystkich dni w zdefiniowanym okresie między $starti $end. Jeśli chcesz dołączyć 10., ustaw$end 11. Możesz dostosować format do swoich potrzeb. Zobacz Podręcznik PHP dla DatePeriod .

Gordon
źródło
2
dobra wiadomość - istnieje łatka do ustawiania flagi zawierającej datę końcową, która (kciuki) sprawi, że będzie w przyszłej wersji.
salathe
8
$begin->setTime(0,0); $end->setTime(12,0);lub inicjowanie z godziną dnia początkowego, ponieważ każda późniejsza niż data końcowa będzie zawierać datę końcową w pętli. Nie jest to najbardziej stylowa poprawka, ale jest to najlepsza opcja, o ile nie ma odpowiedniej flagi.
Chris
31
Jeśli chcesz dołączyć datę końcową do interwału, możesz: $ end = $ end-> modyfikować („+1 dzień”);
JulienITARD
3
Czy można tego użyć, ale odwrócić, aby wrócić do historii?
Jon
3
@JulienITARD to całkiem niezły pomysł, ale bardziej elegancki byłby $ end-> add ($ interwał), ponieważ odpowiada bezpośrednio na zmieniony interwał;)
GDY
92

Dotyczy to również ostatniej daty

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

Jeśli nie potrzebujesz ostatniej daty, po prostu usuń =warunek.

Sabri Aziri
źródło
1
Pamiętaj, że po pętli $beginbędzie inaczej . Ta pętla modyfikuje obiekt utworzony przez new DateTime( "2015-07-03" ). Dlatego powinieneś używać wersji DateTimeImmutable. Ale potrzebujesz dalszych modyfikacji, aby z nich korzystać.
Henk Poley,
40

Konwersja na uniksowe znaczniki czasu ułatwia robienie matematyki dat w 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
}

Jeśli używasz PHP ze strefą czasową z czasem letnim, pamiętaj, aby dodać czas, który nie jest 23:00, 00:00 lub 1:00, aby zabezpieczyć się przed dniami pomijania lub powtarzania.

Harold1983-
źródło
4
Nie podoba mi się wygląd tego 86400. Rozumiem, że to 60 * 60 * 24, ale wciąż ... coś w tym mnie denerwuje.
MikeD,
13
w tym przypadku działa, ale jeśli nastąpi przełączenie między czasem normalnym a czasem oszczędzania światła słonecznego, zakończy się niepowodzeniem, ponieważ istnieje 90000 sekund drugiego dnia, który będziesz miał dwa razy w pętli ...
oezi
2
Mike, najlepiej jest ustawić stałą i nazwać ją „DZIEŃ”, aby stała się o wiele łatwiejsza do odczytania.
The Pixel Developer
5
Będzie to ucierpieć z powodu problemów związanych z czasem letnim. Kiedy przekroczysz punkt czasu letniego, zostanie zepsuty. 12:00 nie jest 12:00 po obu stronach punktu w czasie.
Eric Cope
1
Ten kod ma (każdy kod z 86400 sekundami dziennie) problem z czasem letnim! Z czasem letnim niektóre dni trwają tylko 23 godziny, a około 25 godzin.
sbrbot
20

Skopiuj z próbki php.net dla zakresu obejmującego :

$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>";
}
Alexander Kharchenko
źródło
to najlepsza i najbardziej kompletna odpowiedź. Brakuje tylko jakiegoś wyjaśnienia wartości DateInterval P1D, więc oto kilka przykładów oznaczenia okresu Dwa dni: P2D Dwie sekundy: PT2S Jeden tydzień i dziesięć minut: P1WT10M Y dla lat M dla miesięcy D dla dni W dla tygodni. Są one przeliczane na dni, więc nie można ich łączyć z D. H przez godziny M przez minuty S przez sekundy
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);

lub

$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);
Mark Baker
źródło
2
Wygląda na to, że to rozwiązanie jest wolniejsze niż zaakceptowana odpowiedź (nie uruchomiono niektórych ławek: 100% wolniej przez 60 iteracji). Ale wybrałem ten ze względu na kompatybilność retro dla starych platform hostingowych.
Ifnot
7

Oto kolejna prosta -

/**
 * 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;
}

Przykład:

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
)
HADI
źródło
5

Użyj tej funkcji: -

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

Wywołanie użycia / funkcji: -

Zwiększ o jeden dzień:

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

Wzrost według miesiąca: -

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

użyj trzeciego parametru, jeśli chcesz ustawić format daty: -

   dateRange($start, $end, "+1 month", "Y-m-d H:i:s");//increase by one month and format is mysql datetime
użytkownik2182143
źródło
2

Oto sposób:

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

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

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

     $dtStart->addDay();
 }

Wynikiem $ weekendsInMoth jest szereg dni weekendowych!

Jean Souza
źródło
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"));
}

Możesz również iterować w ten sposób: $_POST['date']Może być wgnieciony z Twojej aplikacji lub witryny Zamiast tego $_POST['date']możesz również umieścić tutaj swój ciąg "21-12-2019". Oba będą działać.

zukayu
źródło
0

Jeśli używasz Laravela i chcesz używać węgla, poprawnym rozwiązaniem byłoby:

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

Pamiętaj, aby dodać:

  • użyj Carbon \ Carbon;
  • użyj Carbon \ CarbonPeriod;
Victor Nuñez
źródło
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)));
    }

?>
Soteris 92
źródło