Ho 2 date in PHP, come posso eseguire un ciclo foreach per passare tutti quei giorni?


207

Sto iniziando con una data 2010-05-01e finendo con 2010-05-10. Come posso scorrere tutte queste date in PHP?

Risposte:


535

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

Questo produrrà tutti i giorni nel periodo definito tra $starte $end. Se si desidera includere il decimo, impostare $endall'undicesimo. Puoi adattare il formato a tuo piacimento. Vedi il manuale di PHP per DatePeriod .


2
buone notizie - esiste una patch per impostare una bandiera per includere la data di fine che (dita incrociate) la trasformerà in una versione futura.
salathe

8
$begin->setTime(0,0); $end->setTime(12,0);o l'inizializzazione con l'ora del giorno della data di inizio, poiché qualsiasi momento successivo a quello della data di fine includerà la data di fine nel ciclo. Non è la soluzione più elegante, ma è l'opzione migliore finché non c'è una bandiera corretta.
Chris,

31
Se vuoi includere la data di fine nel tuo intervallo, puoi fare: $ end = $ end-> modifica ('+1 giorno');
JulienITARD,

3
È possibile usare questo ma invertirlo, per tornare indietro nella storia?
Jon,

3
@JulienITARD è una buona idea, ma più elegante sarebbe $ end-> add ($ intervallo) perché risponde direttamente a un intervallo modificato;)
GDY

92

Questo include anche l'ultima data

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

Se non hai bisogno dell'ultima data, rimuovi =dalla condizione.


1
Assicurati di notare che $beginsarà diverso dopo il loop. Questo ciclo modifica l'oggetto creato da new DateTime( "2015-07-03" ). Ecco perché dovresti usare le versioni DateTimeImmutable. Ma hai bisogno di ulteriori modifiche per usarli.
Henk Poley,

40

La conversione in timestamp unix semplifica la matematica della data in 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
}

Quando si utilizza PHP con un fuso orario con ora legale, assicurarsi di aggiungere un orario diverso dalle 23:00, 00:00 o 1:00 per proteggersi dai giorni saltati o ripetuti.


4
Non mi piace l'aspetto di quell'86400. Capisco che è 60 * 60 * 24, ma comunque ... qualcosa al riguardo mi infastidisce.
MikeD,

13
in questo caso, funziona, ma se c'è un passaggio tra il tempo normale e quello solare, fallirà perché c'è un 90000 di secondo giorno che avrai due volte nel tuo ciclo ...
oezi

2
Mike, la cosa migliore da fare è impostare una costante e chiamarla "DAY" in modo che diventi molto più facile da leggere.
The Pixel Developer

5
Ciò soffrirà di problemi di ora legale. Quando si attraversa un punto dell'ora legale, verrà rovinato. Le 12:00 non sono le 12:00 su entrambi i lati del punto nel tempo.
Eric Cope,

1
Questo codice (ogni codice con 86400 secondi al giorno) ha problemi con l'ora legale! Con l'ora legale alcuni giorni durano solo 23 ore e circa 25 ore.
sbrbot,

20

Copia dall'esempio di php.net per un intervallo inclusivo :

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

questa è la risposta migliore e più completa. Manca solo qualche spiegazione del valore DateInterval P1D, quindi ecco alcuni esempi di Designatore di periodi Due giorni: P2D Due secondi: PT2S Una settimana e dieci minuti: P1WT10M Y per anni M per mesi D per giorni W per settimane. Questi vengono convertiti in giorni, quindi non possono essere combinati con D. H per ore M per minuti S per secondi
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);

o

$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
Sembra che questa soluzione sia più lenta della risposta accettata (un ha eseguito alcuni banchi: 100% più lento per 60 iterazioni). Ma ho scelto questo per retro compatibilità per vecchie forme di hosting.
Se non il

7

Ecco un altro semplice -

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

Esempio:

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

Utente questa funzione: -

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

Chiamata uso / funzione: -

Aumento di un giorno: -

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

Aumento per mese: -

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

utilizzare il terzo parametro se si desidera impostare il formato della data: -

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

2

ecco un modo:

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

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

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

     $dtStart->addDay();
 }

Il risultato di $ weekendInMoth è una serie di giorni di fine settimana!


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

Puoi anche iterare in questo modo, The $_POST['date']può essere ammaccato dalla tua app o dal tuo sito Web Invece di $_POST['date']inserire la stringa qui "21-12-2019". Entrambi funzioneranno.


0

Se si utilizza Laravel e si desidera utilizzare Carbon, la soluzione corretta sarebbe la seguente:

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

Ricorda di aggiungere:

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

?>
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.