20

PHP 中时区的一些问题已经在我脑海中浮现了一段时间,我想知道是否有比我目前正在做的更好的方法来处理它。

所有问题都围绕重新格式化数据库存储日期:

在处理必须支持多个时区(对于用户)的站点时,为了规范存储时间戳的时区,我总是使用CURRENT_TIMESTAMP属性或NOW()函数将它与服务器时区一起存储。

这样我就不必考虑在输入时间戳时为 PHP 设置的时区(因为 PHP 时间函数是时区感知的)。对于每个用户,根据他的偏好,我使用以下命令在引导文件中的某处设置时区:

date_default_timezone_set($timezone);

当我希望使用 phpdate()函数格式化日期时,必须进行某种形式的转换,因为 MySQL 当前以格式存储时间戳Y-m-d H:i:s。不考虑时区,您可以简单地运行:

$date = date($format,strtotime($dbTimestamp));

问题在于它们都是时区感知函数date()strtotime()这意味着如果 PHP 时区的设置与服务器时区不同,则时区偏移量将应用两次(而不是我们想要的一次)。

为了解决这个问题,我通常使用UNIX_TIMESTAMP()不支持时区的函数来检索 MySQL 时间戳,允许我date()直接应用它——从而只应用一次时区偏移量。

我不太喜欢这种“hack”,因为我不能再像往常一样检索这些列,或者*用来获取所有列(有时它大大简化了查询)。此外,有时它根本不是一个使用选项UNIX_TIMESTAMP()(尤其是在使用没有太多抽象查询组合的开源包时)。

另一个问题是在存储时间戳时,当使用CURRENT_TIMESTAMPorNOW()不是一个选项时 - 存储 PHP 生成的时间戳会将它与我想避免的时区偏移一起存储。

我可能在这里遗漏了一些非常基本的东西,但到目前为止,我还没有想出一个通用的解决方案来处理这些问题,所以我不得不逐个处理它们。非常欢迎您的想法

4

4 回答 4

12

几个月前,我们花了一些时间思考这个问题。我们最终得到的技术非常简单:

  1. 以 GMT/UTC 存储日期(例如 0 时区偏移)。
  2. 在从数据库中检索后应用当前用户时区偏移量(例如,在向用户显示之前或您想要的任何时候)。

我们使用 Unix 时间戳格式。但这没关系。

于 2008-12-08T16:12:08.073 回答
7

从 PHP 5.2 开始,您可以使用 DateTime,这使得使用时区变得容易:

$datetime = new DateTime($dbTimestamp, $timezone);
echo $datetime->format('Y-m-d H:i:s');
$datetime->setTimezone(new DateTimeZone('Pacific/Nauru'));
echo $datetime->format('Y-m-d H:i:s');
于 2013-01-30T02:22:28.357 回答
1

您可以尝试强制 MySQL 在任何地方使用 UTC 使用SET time_zone.

不幸的是,我没有得到关于 strtotime/UNIX_TIMESTAMP 的任何答案,事实上我对 Postgres 也有同样的问题。

于 2008-12-06T21:23:39.023 回答
0

我没有在网上找到任何优雅的解决方案,所以我创建了一个时区 HTML 选择生成器脚本,这是直接的输出。是这样的:

<select name="timezone" id="timezone">
    <optgroup label="UTC -11:00">
        <option value="Pacific/Midway">UTC -11:00 Midway</option>
        <option value="Pacific/Niue">UTC -11:00 Niue</option>
        <option value="Pacific/Pago_Pago">UTC -11:00 Pago_Pago</option>
    </optgroup>
    <optgroup label="UTC -10:00">
        <option value="America/Adak">UTC -10:00 Adak</option>
        <option value="Pacific/Honolulu">UTC -10:00 Honolulu</option>
        <option value="Pacific/Johnston">UTC -10:00 Johnston</option>
        <option value="Pacific/Rarotonga">UTC -10:00 Rarotonga</option>
        <option value="Pacific/Tahiti">UTC -10:00 Tahiti</option>
    </optgroup>
    . . . . . . . . . . . . . .
    <optgroup label="UTC +13:00">
        <option value="Pacific/Apia">UTC +13:00 Apia</option>
        <option value="Pacific/Enderbury">UTC +13:00 Enderbury</option>
        <option value="Pacific/Fakaofo">UTC +13:00 Fakaofo</option>
        <option value="Pacific/Tongatapu">UTC +13:00 Tongatapu</option>
    </optgroup>
    <optgroup label="UTC +14:00">
        <option value="Pacific/Kiritimati">UTC +14:00 Kiritimati</option>
    </optgroup>
</select>

享受!

于 2013-06-10T20:44:17.230 回答