20

您如何验证 ISO 8601 日期字符串(例如:2011-10-02T23:25:42Z)。

我知道 ISO 8601 日期有几种可能的表示形式,但我只对验证上面作为示例给出的格式感兴趣。

谢谢!

4

6 回答 6

18

这对我有用,它使用正则表达式来确保日期是您想要的格式,然后尝试解析日期并重新创建它以确保输出与输入匹配:

<?php

$date = '2011-10-02T23:25:42Z';
var_dump(validateDate($date));

$date = '2011-17-17T23:25:42Z';
var_dump(validateDate($date));

function validateDate($date)
{
    if (preg_match('/^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2})Z$/', $date, $parts) == true) {
        $time = gmmktime($parts[4], $parts[5], $parts[6], $parts[2], $parts[3], $parts[1]);

        $input_time = strtotime($date);
        if ($input_time === false) return false;

        return $input_time == $time;
    } else {
        return false;
    }
}

您可以进一步扩展以使用checkdate来确保月份和年份也是有效的。

于 2011-11-04T00:54:28.530 回答
16

编辑:到目前为止,最简单的方法是简单地尝试DateTime使用字符串创建一个对象,例如

$dt = new DateTime($dateTimeString);

如果DateTime构造函数无法解析字符串,则会抛出异常,例如

DateTime::__construct():无法在位置 18 (2) 解析时间字符串 (2011-10-02T23:25:72Z):意外字符

请注意,如果您不使用时区指示符,它将使用配置的默认时区。

第二个最简单的方法是使用正则表达式。像这样的东西可以掩盖它

if (preg_match('/^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2})(Z|(\+|-)\d{2}(:?\d{2})?)$/', $dateString, $parts)) {
    // valid string format, can now check parts

    $year  = $parts[1];
    $month = $parts[2];
    $day   = $parts[3];

    // etc
}
于 2011-11-04T00:54:32.227 回答
15

我想分享我的轻量级解决方案。这并不理想,但可能对某人有所帮助。

function validISO8601Date($value)
{
    if (!is_string($value)) {
        return false;
    }

    $dateTime = \DateTime::createFromFormat(\DateTime::ISO8601, $value);

    if ($dateTime) {
        return $dateTime->format(\DateTime::ISO8601) === $value;
    }

    return false;
}

注意!

一些有效的 ISO8601 日期将失败查看下面的列表

NOT VALID  --> '' // Correct
NOT VALID  --> 'string' // Correct
VALID      --> '2000-01-01T01:00:00+1200' // This is the only format function returns as valid
NOT VALID  --> '2015-01 first' // Correct
NOT VALID  --> '2000-01-01T01:00:00Z' // Must be valid!
NOT VALID  --> '2000-01-01T01:00:00+01' // Must be valid!
于 2014-01-21T14:19:29.670 回答
6

请参阅http://www.pelagodesign.com/blog/2009/05/20/iso-8601-date-validation-that-doesnt-suck/。它让这个正则表达式可以使用:

^([\+-]?\d{4}(?!\d{2}\b))((-?)((0[1-9]|1[0-2])(\3([12]\d|0[1-9]|3[01]))?|W([0-4]\d|5[0-2])(-?[1-7])?|(00[1-9]|0[1-9]\d|[12]\d{2}|3([0-5]\d|6[1-6])))([T\s]((([01]\d|2[0-3])((:?)[0-5]\d)?|24\:?00)([\.,]\d+(?!:))?)?(\17[0-5]\d([\.,]\d+)?)?([zZ]|([\+-])([01]\d|2[0-3]):?([0-5]\d)?)?)?)?$

我想这并不能完全回答你的问题,因为它会匹配任何有效的 ISO 8601 日期,但如果没问题,那么它就完美了。

于 2011-11-04T00:55:46.097 回答
3

这是我使用的功能。它类似于 draw010 的答案,但也适用于以“+01:00”或“-01:00”结尾的时间戳。

function isTimestampIsoValid($timestamp)
{
    if (preg_match('/^'.
            '(\d{4})-(\d{2})-(\d{2})T'. // YYYY-MM-DDT ex: 2014-01-01T
            '(\d{2}):(\d{2}):(\d{2})'.  // HH-MM-SS  ex: 17:00:00
            '(Z|((-|\+)\d{2}:\d{2}))'.  // Z or +01:00 or -01:00
            '$/', $timestamp, $parts) == true)
    {
        try {
            new \DateTime($timestamp);
            return true;
        }
        catch ( \Exception $e)
        {
            return false;
        }
    } else {
        return false;
    }
}
于 2014-01-21T09:18:16.947 回答
0

Carbon 对这部分的处理非常好

use Carbon\Carbon;
use Carbon\Exceptions\InvalidFormatException;
...
/** @test */
public function checking_iso8601_date()
{
    $this->expectException(InvalidFormatException::class);
    Carbon::createFromFormat('c', '2021-05-11 20:03:45+02:00');

    $this->assertInstanceOf(Carbon::class, Carbon::createFromFormat('c', '2021-05-11T20:03:45+02:00'));
}

你可以对 DateTime 做同样的事情

use DateTime;
...
/** @test */
public function checking_iso8601_date()
{
    $this->assertInstanceOf(DateTime::class, DateTime::createFromFormat('Y-m-d\TH:i:sP', '2021-05-11T20:03:45+02:00'));
    $this->assertFalse(DateTime::createFromFormat('Y-m-d\TH:i:sP', '2021-05-11 20:03:45+02:00'));
}
于 2021-05-11T18:22:20.800 回答