Eu tenho duas datas no PHP, como posso executar um loop foreach para passar por todos esses dias?

207

Estou começando com uma data 2010-05-01e terminando com 2010-05-10. Como posso iterar todas essas datas no PHP?

Shamoon
fonte

Respostas:

535

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

Isso produzirá todos os dias no período definido entre $starte $end. Se você deseja incluir o 10º, defina $endcomo 11º. Você pode ajustar o formato ao seu gosto. Veja o Manual do PHP para DatePeriod .

Gordon
fonte
2
boas notícias - existe um patch para definir uma bandeira para incluir a data final que (dedos cruzados) a transformará em uma versão futura.
Salathe
8
$begin->setTime(0,0); $end->setTime(12,0);ou inicializando com a hora do dia da data de início, pois qualquer hora posterior à data de término incluirá a data de término no loop. Não é a solução mais elegante, mas é a melhor opção, desde que não haja uma bandeira adequada.
Chris
31
Se você deseja incluir a data final no seu intervalo, você pode: $ end = $ end-> modify ('+1 day');
perfil completo de JulienITARD
3
É possível usar isso, mas invertê-lo, para voltar à história?
Jon
3
@JulienITARD isso é uma boa idéia, mas mais elegante seria de R $ final de> add ($ intervalo) porque responde diretamente para um intervalo alterado;)
GDY
92

Isso também inclui a última 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 você não precisar da última data, remova =-a da condição.

Sabri Aziri
fonte
1
Certifique-se de observar que $beginserá diferente após o loop. Este loop modifica o objeto criado por new DateTime( "2015-07-03" ). Por isso, você deve usar as versões DateTimeImmutable. Mas você precisa de mais algumas modificações para usá-las.
Henk Poley
40

A conversão para timestamps unix facilita a matemática de datas no 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
}

Ao usar o PHP com um fuso horário com horário de verão, certifique-se de adicionar um horário que não seja 23:00, 00:00 ou 1:00 para se proteger contra dias pulados ou repetidos.

Harold1983-
fonte
4
Não gosto da aparência desse 86400. Entendo que é 60 * 60 * 24, mas ainda assim ... algo sobre isso me incomoda.
8119 MikeD
13
neste caso, ele funciona, mas se houver um interruptor entre a economia de tempo normal e luz solar, ele irá falhar porque não há uma segunda-dia 90000 que você vai ter duas vezes em seu ciclo ...
oezi
2
Mike, a melhor coisa a fazer é configurar uma constante e nomeá-la como "DIA", para que fique muito mais fácil ler.
The Pixel Developer
5
Isso sofrerá com problemas de horário de verão. Quando você cruza um ponto de horário de verão, ele fica ferrado. 12:00 não é 12:00 em ambos os lados do ponto no tempo.
Eric Cope
1
Esse código (cada código com 86400 segundos por dia) tem problemas com o horário de verão! Com o horário de verão, alguns dias duram apenas 23 horas e 25 horas.
Sbrbot
20

Copie da amostra php.net para um intervalo 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>";
}
Alexander Kharchenko
fonte
esta é a melhor e mais completa resposta. Faltam apenas algumas explicações do valor DateInterval P1D, portanto, aqui estão alguns exemplos de designador de período Dois dias: P2D Dois segundos: PT2S Uma semana e dez minutos: P1WT10M Y nos anos M nos meses D nos dias W nas semanas. Estes são convertidos em dias, portanto, não podem ser combinados com D. H por horas M por minutos S por segundos
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);

ou

$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
fonte
2
Parece que esta solução é mais lenta do que a resposta aceita (alguns bancos foram executados: 100% mais lento para 60 iterações). Mas eu escolhi este para compatibilidade retroativa com placas de hospedagem antigas.
Caso contrário,
7

Aqui está outro simples -

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

Exemplo:

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
fonte
5

Utiliza esta função: -

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

Chamada de uso / função: -

Aumentar em um dia: -

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

Aumento por mês: -

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

use o terceiro parâmetro se desejar definir o formato da data: -

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

aqui está uma maneira:

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

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

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

     $dtStart->addDay();
 }

O resultado de $ weekendsInMoth é uma variedade de dias de fim de semana!

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

Você também pode fazer a iteração, O $_POST['date']aplicativo pode ser amassado no seu aplicativo ou site. Em vez de $_POST['date']você também pode colocar sua string aqui "21-12-2019". Ambos irão funcionar.

zukayu
fonte
0

Se você usa o Laravel e deseja usar o Carbon, a solução correta seria a seguinte:

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

Lembre-se de adicionar:

  • use Carbono \ Carbono;
  • use Carbon \ CarbonPeriod;
Victor Nuñez
fonte
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
fonte