使用PHP处理DST的时区

使用PHP处理DST的时区,php,datetime,utc,dst,timezone-offset,Php,Datetime,Utc,Dst,Timezone Offset,我正在开发一个日历应用程序。在这种情况下,来自不同时区的用户创建/查看事件。 我想按照下面的方法以UTC格式保存事件时间,并以用户的本地时间显示它们。注意:用户有其首选的时区设置 将事件开始时间保存为UTC时间戳: $timezone = new DateTimeZone( $user_timezone_name ); $utcOffset = $timezone->getOffset( new DateTime( $event_start_time_string_local ) ); $

我正在开发一个日历应用程序。在这种情况下,来自不同时区的用户创建/查看事件。 我想按照下面的方法以UTC格式保存事件时间,并以用户的本地时间显示它们。注意:用户有其首选的时区设置

将事件开始时间保存为UTC时间戳:

$timezone = new DateTimeZone( $user_timezone_name );
$utcOffset = $timezone->getOffset( new DateTime( $event_start_time_string_local ) );
$event_start_timestamp_local = maketime( $event_start_time_string_local );
//Now add/subtract $utcOffset to/from $event_start_timestamp_local to get event's start  //time  timestamp in UTC and save this result in DB.
date_default_timezone_set( $user_timezone_name );
$eventTimeLocalTime = date("Y-m-d H:i:s", $event_start_timestamp_in_UTC );
获取用户时区中的事件开始时间:

$timezone = new DateTimeZone( $user_timezone_name );
$utcOffset = $timezone->getOffset( new DateTime( $event_start_time_string_local ) );
$event_start_timestamp_local = maketime( $event_start_time_string_local );
//Now add/subtract $utcOffset to/from $event_start_timestamp_local to get event's start  //time  timestamp in UTC and save this result in DB.
date_default_timezone_set( $user_timezone_name );
$eventTimeLocalTime = date("Y-m-d H:i:s", $event_start_timestamp_in_UTC );
其中:

user_timezone_name is user's timezone setting.
event_start_time_string_local is event's start time string in local/civil time.
event_start_timestamp_in_UTC is event's start time timestamp in UTC.
我的问题:

  • 上面使用的PHP API是否负责所有区域的DST
  • DST时间本身在不同的年份会发生变化,PHP是如何获得这方面的信息的?我们需要升级PHP吗?如果是这样,我们需要升级全部或特定的PHP库吗
参考资料: -
-你把这件事搞得太复杂了。要使用
DateTime
在两个时区之间转换,请执行以下操作:

date_default_timezone_set('Asia/Kolkata'); // YOUR timezone, of the server

$date = new DateTime($input, new DateTimeZone('Asia/Tokyo')); // USER's timezone
$date->setTimezone(new DateTimeZone('UTC'));
echo $date->format('Y-m-d H:i:s');
这将从用户的本地时区转换为UTC。反过来说,要在用户的本地时间中显示时间,请交换两个时区


是的,PHP负责DST。必要的转换规则是PHP安装的一部分。您可以通过更新PHP或更新。是的,PHP api考虑了DST。大多数情况下,您可以按照deceze显示的方式安全地执行时间转换。问题是它并不总是可能/可行或有意义的。考虑下面的例子-在波兰(在2013)在10月27日凌晨3点,你的时钟被移动回一个小时,切换到标准时间(偏移UTC+01:00),3月31日在2AM时钟被调整1H,切换到夏季时间(偏移UTC+02:00)。现在,让我们看看Grenwich/UTC中的“实时”滴答声,以及PHP从欧洲/华沙到UTC的转换是如何工作的:

Europe/Warsaw time   offset  UTC                    php2utc conversion   php offset
2013-10-27 01:00:00    +2    2013-10-26 23:00:00    2013-10-26 23:00:00  +2
2013-10-27 01:30:00    +2    2013-10-26 23:30:00    2013-10-26 23:30:00  +2
2013-10-27 02:00:00    +2    2013-10-27 00:00:00    2013-10-27 01:00:00  +1  *
2013-10-27 02:30:00    +2    2013-10-27 00:30:00    2013-10-27 01:30:00  +1  *
2013-10-27 02:59:00    +2    2013-10-27 00:59:00    2013-10-27 01:59:00  +1  *
     3am -> 2am .....................................summer time changes to standard(winter) time @3am we subtract 1h so 3am becomes 2am
2013-10-27 02:00:00    +1    2013-10-27 01:00:00    2013-10-27 01:00:00  +1
2013-10-27 02:30:00    +1    2013-10-27 01:30:00    2013-10-27 01:30:00  +1
2013-10-27 03:00:00    +1    2013-10-27 02:00:00    2013-10-27 02:00:00  +1  
2013-10-27 03:30:00    +1    2013-10-27 02:30:00    2013-10-27 02:30:00  +1  
如您所见,在凌晨2点到凌晨3点之间,PHP无法知道“哪一个”小时(从调整前/调整后开始),因此无法可靠地进行转换。
希望这有助于某人的思考;-)

更新时区B是一项常规任务,每年需要几次,因为每次任何国家进行更改都会得到更新,而且这种情况非常频繁。为什么我们需要称之为“日期默认时区集”(“亚洲/加尔各答”)“开始时?您可以使用
”@12345…“
作为UNIX时间戳。@Krishna UNIX时间戳定义为UTC 1970年以来的秒数。时间戳是无时区的,但如果有什么区别的话,它是UTC。您可以将中的格式设置为如上所示的“本地”时间。@deceze UNIX时间戳定义为自历元以来的“UNIX秒”数。UNIX秒!=s、 i.秒,因为unix秒包括闰秒,因此所有天的长度正好是86400秒,而不是86400.002秒的平均长度。一个结果是,闰秒的时刻不能在unix时间戳中表示,因为它们不存在于unix时间戳中。这正是我试图编写单元测试的问题。DateTimeZone对象可能包括DST更改。我怎样才能强迫它这样或那样呢?是否有一些一致性(例如,如果我以小时为单位指定一个时间,重叠,它将始终选择DST时间),以及缺少的小时又如何。在上述示例中,波兰没有当地时间3月31日凌晨2:30(即UTC+2的凌晨3:30)。大概在某个地方有详细的规范。知道在哪里吗?谢谢你的桌子。在这一小时内,无法输入CEST值,因为它们总是作为CET处理。即使明确给出了CEST时区。我通过检测DST变化(使用
格式('I')
)解决了这个问题,在这种情况下,我创建了一个提前一小时的日期时间,将其转换为UTC,然后添加丢失的小时。现在,我的UTC数据库的小时值没有漏洞或重复的时间戳。