>我在将日期时间从莫斯科时区转换为纽约时区时遇到问题。这是我的测试脚本:
$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;
}