PHP 日期时间转换问题


PHP DateTime converting issue

>我在将日期时间从莫斯科时区转换为纽约时区时遇到问题。这是我的测试脚本:

$year = 2015;
$month = 3;
$tzMoscow = new DateTimeZone('Europe/Moscow');
$tzNewYork = new DateTimeZone('America/New_York');
$startDate = DateTime::createFromFormat('Y-n-d', "$year-$month-01", $tzMoscow);
echo $startDate->format('Y-m-d H:i:s')."'n"; // 2015-03-01 16:16:05
$startDate = DateTime::createFromFormat('Y-n-d', "$year-$month-01", $tzNewYork);
echo $startDate->format('Y-m-d H:i:s') . "'n"; // 2015-03-01 09:16:05
$startDate->setTimezone($tzMoscow);
echo $startDate->format('Y-m-d H:i:s') . "'n"; // 2015-03-01 17:16:05

第三个输出不正确,时间应为 16:16:05。我做错了什么还是这是 php 中的错误?

我想我已经明白了。问题是您没有指定一天中的时间,因此createFromFormat使用的是"当前系统时间":

如果 format 不包含字符 !,则未在格式中指定的生成时间部分将设置为当前系统时间。

现在,当该时区在指定日期(3 月 1 日)和当前日期(3 月 14 日)之间更改了 UTC 偏移量时,使用"当前系统时间"在纽约时区中创建时间是什么意思?3 月 1 日是 UTC-5,由于夏令时,现在是 UTC-4。

我相信 PHP 正在获取指定时区的当前时间(您运行该代码时为 9:16),然后将其用作指定日期的一天中的时间。因此,我们最终会得到 2015 年 3 月 1 日,纽约时间 09:16 - 或 UTC 时间 3 月 1 日 14:16,这确实是莫斯科时间 3 月 1 日 17:16。

这与您运行代码时莫斯科的当前时间不同,即 16:16。

基本上,你应该尽量不要这样做 - 或者期望这样的问题发生。想想你真正想要表示一天中的什么时间,请记住,在特定时区内,偏移量会随着时间的推移而变化。我真的不能建议你的代码应该是什么,因为我们不知道你想实现什么 - 但是使用一天中的当前时间作为不同的日期肯定会导致这个问题。

我也相信这是夏令时的问题,3 月 8 日纽约时区发生变化。我将您的日期格式扩展到Y-m-d H:i:s U I以包括 unix 时间戳和 DST 值。

输出现在如下所示:

2015-03-01 19:07:17 1425222437 0
2015-03-01 11:07:17 1425226037 0
2015-03-01 20:07:17 1425226037 0

如您所见,创建的DateTime对象之间的 unix 时间戳已经不同。

现在,当我将具体时间指定为

$startDate = DateTime::createFromFormat('Y-n-d H:i', "$year-$month-01 00:00", $tzMoscow);

以及

$startDate = DateTime::createFromFormat('Y-n-d H:i', "$year-$month-01 00:00", $tzNewYork);

输出更改为:

2015-03-01 00:00:00 1425153600 0
2015-03-01 00:00:00 1425186000 0
2015-03-01 09:00:00 1425186000 0

另一方面,如果我将$month更改为4(纽约使用 DST),我会得到以下输出:

2015-04-01 19:08:35 1427900915 0
2015-04-01 11:08:35 1427900915 1
2015-04-01 19:08:35 1427900915 0

这些结果意味着什么?

纽约和莫斯科时区之间的转换在所有情况下都正确,因为您可以从 unix 时间戳是否相同来判断。此外,"2015-03-01 00:00"显然是莫斯科和纽约的不同时间戳,因为它们取决于具体时区。

所以我认为你的代码是正确的,php 中没有错误。但是,由于 3 月 1 日至今天(3 月 14 日)之间的 DST 切换,纽约和莫斯科的"当前"时间有所不同。

因此,虽然乔恩的回答已经解释了这个理论(我不想要所有的功劳,他是第一个),也许仍然有人会发现一些具体的例子很有用。

您可以使用

此功能在时区之间切换

function changeTimezone($time, $currentTimezone, $timezoneRequired, $FormtsTime = 'Y-m-d h:i:s')
{
    $dayLightFlag = false;
    $dayLgtSecCurrent = $dayLgtSecReq = 0;
    $system_timezone = date_default_timezone_get();
    $local_timezone = $currentTimezone;
    date_default_timezone_set($local_timezone);
    $local = date($FormtsTime);
    date_default_timezone_set("GMT");
    $gmt = date($FormtsTime);
    $require_timezone = $timezoneRequired;
    date_default_timezone_set($require_timezone);
    $required = date($FormtsTime);
    date_default_timezone_set($system_timezone);
    $diff1 = (strtotime($gmt) - strtotime($local));
    $diff2 = (strtotime($required) - strtotime($gmt));
    $date = new DateTime($time);
    $date->modify("+$diff1 seconds");
    $date->modify("+$diff2 seconds");
    if ($dayLightFlag) {
        $final_diff = $dayLgtSecCurrent + $dayLgtSecReq;
        $date->modify("$final_diff seconds");
    }
    $timestamp = $date->format($FormtsTime);
    return $timestamp;
}