web-dev-qa-db-fra.com

PHP DateTime :: createFromFormat n'analyse pas l'heure date ISO 8601

Code parle un million de mots:

php > echo strtotime("2010-12-07T23:00:00.000Z");
1291762800
echo date('c', 1291762800);
2010-12-08T00:00:00+01:00
php > var_dump(DateTime::createFromFormat('c', "2010-12-07T23:00:00.000Z"));
bool(false)
php > var_dump(DateTime::createFromFormat(DateTime::ISO8601, "2010-12-07T23:00:00.000Z"));
bool(false)

Une idée de ce qui se passe? 

Au fait, oui, le nouveau DateTime ("2010-12-07T23: 00: 00.000Z") fonctionne bien. Mais je préfère savoir quelle entrée je reçois. 

39
Jake

Il y a un rapport de bogue qui décrit votre problème exactement :)

https://bugs.php.net/bug.php?id=51950

Depuis le 2016-08-07, le rapport de bogue a été marqué comme "pas un bogue". Vous devez utiliser strtotime ou new DateTime à la place.

Les constantes définies s'appliquent à la fois au formatage et à l'analyse syntaxique, ce qui force votre chemin.

36
Ja͢ck

Analyse de la date ISO8601 et changement de fuseau horaire:

// create ISO8601 dateTime 
$date = DateTime::createFromFormat(DateTime::ISO8601, '2016-07-27T19:30:00Z');

// set to user's timezone
$date -> setTimeZone('Asia/Singapore');

echo $date -> format(DateTime::ISO8601);
// prints '2016-07-28T03:30:00+0800'
13
a20

Personne n’a mentionné l’utilisation de DATE_ATOM, qui, autant que je sache, est l’implémentation la plus correcte de la norme ISO 8601. Cela devrait au moins fonctionner pour les 3 derniers:

<?php

$dates = array(
    "2010-12-07T23:00:00.000Z",
    "2010-12-07T23:00:00",
    "2010-12-07T23:00:00Z",
    "2010-12-07T23:00:00+01:00",
    (new \DateTime("now"))->format(DATE_ATOM)
);

foreach($dates as $d) {

    $res = \DateTime::createFromFormat(DATE_ATOM, $d);

    echo "try $d: \n";
    var_dump($res);
    echo "\n\n";
}

?>

Pour pouvoir tous les analyser, j'ai écrit une petite fonction:

<?php

function parse_iso_8601($iso_8601_string) {
    $results = array();
    $results[] = \DateTime::createFromFormat("Y-m-d\TH:i:s",$iso_8601_string);
    $results[] = \DateTime::createFromFormat("Y-m-d\TH:i:s.u",$iso_8601_string);
    $results[] = \DateTime::createFromFormat("Y-m-d\TH:i:s.uP",$iso_8601_string);
    $results[] = \DateTime::createFromFormat("Y-m-d\TH:i:sP",$iso_8601_string);
    $results[] = \DateTime::createFromFormat(DATE_ATOM,$iso_8601_string);

    $success = array_values(array_filter($results));
    if(count($success) > 0) {
        return $success[0];
    }
    return false;
}

// Test
$dates = array(
    "2010-12-07T23:00:00.000Z",
    "2010-12-07T23:00:00",
    "2010-12-07T23:00:00Z",
    "2010-12-07T23:00:00+01:00",
    (new \DateTime("now"))->format(DATE_ATOM)
);

foreach($dates as $d) {

    $res = parse_iso_8601($d);

    echo "try $d: \n";
    var_dump($res);
    echo "\n\n";
}

?>

Comme @Glutexo l'a mentionné, cela ne fonctionne que s'il n'y a que 1 à 6 chiffres de précision pour la partie décimale également. N'hésitez pas à l'améliorer.

8
steven

essaye ça:

DateTime::createFromFormat('Y-m-d\TH:i:sP', $date)
6
Etienne

Utilisez DATE_ATOM plutôt que 'c' lorsque vous formatez comme @Steven. Voici comment vous travaillez avec ISO 8601 en PHP.

<?php
$now_date = new DateTime();
$now_iso_8601 = $now_date->format(DATE_ATOM);
echo "Now in ISO 8601 format: {$now_iso_8601}\n";
$date_from_string_and_format = date_create_from_format(DATE_ATOM, $now_iso_8601);
echo "ISO 8601 formatted string, back to DateTime object:\n";
var_dump($date_from_string_and_format);

empreintes

Now in ISO 8601 format: 2018-09-05T08:17:35-10:00
ISO 8601 formatted string, back to DateTime object:
object(DateTime)#2 (3) {
  ["date"]=>
  string(26) "2018-09-05 08:17:35.000000"
  ["timezone_type"]=>
  int(1)
  ["timezone"]=>
  string(6) "-10:00"
}
2
John Erck

Pour la réponse indiquée ici https://stackoverflow.com/a/14849503/2425651 Nous pouvons utiliser ce format "Y-m-d\TH: i: s.u +" pour conserver les microsecondes.

$format = 'Y-m-d\TH:i:s.u+';
$value = '2017-09-21T10:11:19.026Z'; // jsDate.toUTCString();
var_dump(\DateTime::createFromFormat($format, $value));
2
Dzung Nguyen

Il est très étrange et décevant que ce bogue soit toujours d'actualité ..__ Voici un bon modèle pour analyser la date avec des microsecondes en décimales décimales:

Y-m-d\TH:i:s.uO

Usage:

$dateStr = '2015-04-29T11:42:56.000+0400'
$ISO = 'Y-m-d\TH:i:s.uO'
$date = DateTime::createFromFormat($ISO, $dateStr)
2
Dmitry Davydov

Simplement:

$dt = new DateTime('2018-04-07T16:32:44Z');
$dt->format('Ymd'); // 20180407
2
stloc

Celui-ci fonctionne pour moi:

$date = (new DateTime)->setTimestamp(strtotime('2017-12-31T23:00:00.000Z'));
0
rckd