Преобразование смещения UTC в часовой пояс или дату

скребок для головы для вас.

я захватываю данные geo IP от IPInfoDBAPI и возвращает смещение часового пояса от UTC в том числе DST (если в настоящее время отражено).

например, я живу в EST (-5), и в настоящее время это DST, поэтому GEO IP API возвращает (-04:00) в качестве смещения.

это замечательно, так как DST-это чертова головная боль. Но, к моему удивлению, это вызвало еще одну головную боль.

загрузить эти данные в PHP должны быть переданы через AJAX в приложение. Я хотел бы иметь живое местное время IP-адреса в приложении.

у меня все это отлично настроено, но я схожу с ума, пытаясь выяснить, как установить часовой пояс PHP в соответствии со смещением, чтобы я мог просто захватить текущие часы date('H'); и протокол date('i'); перейти на via AJAX.

Я не уверен, есть ли конкретная функция, которая может просто дать мне текущие часы и минуты на основе этого смещения или если есть практический способ установить часовой пояс на основе смещения (который будет иметь DST уже применяется, если он действует).

я искал и искал Google, чтобы найти ответ на этот вопрос, но то, что я делаю, более конкретно, так как DST уже применяется.

я нашел одну функцию на PHP.net это, кажется, делает трюк (он работает для моего часового пояса и возвращает правильное время), хотя для других часовых поясов, таких как PST, он возвращается через 1 час чем это должно быть, даже если смещение правильное (-07:00 переход на летнее время).

часовой пояс, возвращаемый из функции Chile/EasterIsland у меня такое чувство, что это причина. Если бы я мог, я бы сделал это только для США, но мне нужно, чтобы это было во всем мире.

это функция, которую я имею сейчас. Пожалуйста, извините за очень грязный код. За последние несколько часов я играл с кучей вещей, пытаясь найти решение.

большая часть функциональность была найдена в интернете.

function offsetToTZ($offset) {
switch((string) $offset) {
    case '-04:30' : return 'America/Caracas'; break;
    case '-03:30' : return 'Canada/Newfoundland'; break;
    case '+03:30' : return 'Asia/Tehran'; break;
    case '+04:30' : return 'Asia/Kabul'; break;
    case '+05:30' : return 'Asia/Kolkata'; break;
    case '+05:45' : return 'Asia/Kathmandu'; break;
    case '+09:30' : return 'Australia/Darwin'; break;
}
$offset = (int) str_replace(array('0',0,':00',00,'30',30,'45',45,':','+'),'', (string) $offset);

$offset = $offset*60*60;
$abbrarray = timezone_abbreviations_list(); 
foreach ($abbrarray as $abbr) { 
    foreach($abbr as $city) { 
        if($city['offset'] == $offset) { 
            return $city['timezone_id'];
        }
    }
}
return false; 
}

я включил переключатель / корпус для определенных часовых поясов, которые :30 и :45 там. Может быть способ включить это также без необходимости переключателя / корпуса.

Примечание: смещения всегда возвращаются как таковые +00:00 или -00:00 из API geo IP.

Я признателен за любую помощь или точку в правильном направлении. Я не очень новичок в PHP, но смещения-это новая история для мне. Спасибо!

8 ответов


это можно сделать довольно просто, повернув смещение в секунды и передав его timezone_name_from_abbr:

<?php
$offset = '-7:00';

// Calculate seconds from offset
list($hours, $minutes) = explode(':', $offset);
$seconds = $hours * 60 * 60 + $minutes * 60;
// Get timezone name from seconds
$tz = timezone_name_from_abbr('', $seconds, 1);
// Workaround for bug #44780
if($tz === false) $tz = timezone_name_from_abbr('', $seconds, 0);
// Set timezone
date_default_timezone_set($tz);

echo $tz . ': ' . date('r');

демо

третий параметр timezone_name_from_abbr определяет, будет ли переход на летнее время или нет.

ошибка #44780:

timezone_name_from_abbr () вернет false в некотором часовом поясе смещения. В частности-Гавайи, которые имеют -10 от GMT смещения, -36000 считанные секунды.

ссылки:


date_default_timezone_set('UTC');

$timezones = array();
foreach (DateTimeZone::listAbbreviations() as $key => $array)
{
    $timezones = array_merge($timezones, $array);
}

$utc                = new DateTimeZone('UTC');
$timezone_offset    = '+02:00'; # 2H
$sign               = substr($timezone_offset, 0, 1) == '+'? '': '-';
$offset             = substr($timezone_offset, 1, 2) . 'H' . substr($timezone_offset, 4, 2) . 'M';

$operation = $sign == ''? 'add': 'sub';

$start  = new DateTime('', $utc);
$date   = new DateTime('', $utc);

$date->{$operation}(new DateInterval("PT{$offset}"));

$offset = $start->diff($date)->format('%r') . ($start->diff($date)->h * 3600 + $start->diff($date)->m * 60 + $start->diff($date)->s); # 7200 (2H)

echo $offset, PHP_EOL;
echo $date->format('Y-m-d H:i:s'), PHP_EOL;

foreach($timezones as $timezone)
{
    if($timezone['offset'] == $offset)
    {
        echo $timezone['timezone_id'], PHP_EOL;
    }
}

возможно, я неправильно понял вас в некоторых частях, но я надеюсь, что это поможет, если бы вы могли быть более конкретными, я мог бы быть более полезным.

для Чили я получаю:

-25200 (-7h)
2012-08-07 18:05:24 (current time 2012-08-08 01:05:24)
Chile/EasterIsland

вывод примера выше:

7200
2012-08-08 02:49:56
Europe/London
Europe/Belfast
Europe/Gibraltar
Europe/Guernsey
Europe/Isle_of_Man
Europe/Jersey
GB
Africa/Khartoum
Africa/Blantyre
Africa/Bujumbura
Africa/Gaborone
Africa/Harare
Africa/Kigali
Africa/Lubumbashi
Africa/Lusaka
Africa/Maputo
Africa/Windhoek
Europe/Berlin
Africa/Algiers
Africa/Ceuta
Africa/Tripoli
Africa/Tunis
Arctic/Longyearbyen
Atlantic/Jan_Mayen
CET
Europe/Amsterdam
Europe/Andorra
Europe/Athens
Europe/Belgrade
Europe/Bratislava
Europe/Brussels
Europe/Budapest
Europe/Chisinau
Europe/Copenhagen
Europe/Gibraltar
Europe/Kaliningrad
Europe/Kiev
Europe/Lisbon
Europe/Ljubljana
Europe/Luxembourg
Europe/Madrid
Europe/Malta
Europe/Minsk
Europe/Monaco
Europe/Oslo
Europe/Paris
Europe/Podgorica
Europe/Prague
Europe/Riga
Europe/Rome
Europe/San_Marino
Europe/Sarajevo
Europe/Simferopol
Europe/Skopje
Europe/Sofia
Europe/Stockholm
Europe/Tallinn
Europe/Tirane
Europe/Tiraspol
Europe/Uzhgorod
Europe/Vaduz
Europe/Vatican
Europe/Vienna
Europe/Vilnius
Europe/Warsaw
Europe/Zagreb
Europe/Zaporozhye
Europe/Zurich
WET
Europe/Kaliningrad
Europe/Helsinki
Africa/Cairo
Africa/Tripoli
Asia/Amman
Asia/Beirut
Asia/Damascus
Asia/Gaza
Asia/Istanbul
Asia/Nicosia
EET
Europe/Athens
Europe/Bucharest
Europe/Chisinau
Europe/Istanbul
Europe/Kaliningrad
Europe/Kiev
Europe/Mariehamn
Europe/Minsk
Europe/Moscow
Europe/Nicosia
Europe/Riga
Europe/Simferopol
Europe/Sofia
Europe/Tallinn
Europe/Tiraspol
Europe/Uzhgorod
Europe/Vilnius
Europe/Warsaw
Europe/Zaporozhye
Asia/Jerusalem
Asia/Gaza
Asia/Tel_Aviv
MET
Africa/Johannesburg
Africa/Maseru
Africa/Mbabane
Africa/Windhoek
Africa/Windhoek
Africa/Ndjamena
Europe/Lisbon
Europe/Madrid
Europe/Monaco
Europe/Paris
WET
Europe/Luxembourg

который гвозди мой часовой пояс.


$UTC_offset = '+03:00';
$date       = new \DateTime('now', 'UTC');
var_dump($date);
$timezone  = new \DateTimeZone(str_replace(':', '', $UTC_offset));
$date->setTimezone($timezone);
var_dump($date);

результаты:

class DateTime#205 (3) {
  public $date =>
  string(26) "2015-01-20 06:00:00.000000"
  public $timezone_type =>
  int(3)
  public $timezone =>
  string(3) "UTC"
}
class DateTime#205 (3) {
  public $date =>
  string(26) "2015-01-20 09:00:00.000000"
  public $timezone_type =>
  int(1)
  public $timezone =>
  string(6) "+03:00"
}

возможно, вы захотите взглянуть на DateTime расширение PHP (оно включено и включено по умолчанию во всех версиях PHP >= 5.2.0, если оно не было специально отключено во время компиляции).

Он делает все, что вам нужно здесь довольно хорошо.


не рекомендуется сопоставлять смещения часового пояса с идентификатором часового пояса. Многие часовые пояса имеют одинаковое смещение.

даже в пределах одной страны это проблематично. Учтите, что в Соединенных Штатах смещение -5 используется как Центральным стандартным временем (America/Chicago) и Восточное летнее время (America/New_York) в разное время года.

также учтите, что есть моменты, когда оба этих часовых пояса используют -5 одновременно. Например, воскресенье ноября 3rd 2013 в 1: 00 AM в UTC-5 может быть либо в CDT, либо в EST. У вас не было бы возможности отличить только на -5, какой часовой пояс это был.

подробности здесь.


вы можете использовать GMT время также и преобразовать его в ваше требование после

<?php
echo gmdate("M d Y H:i:s", mktime(0, 0, 0, 1, 1, 1998));
?>

GMT означает среднее время по Гринвичу, которое распространено во всем мире.


в настоящее время DateTimeZone конструктор может явно принять смещение UTC, которое, как я понимаю, у вас есть.

Так же:

$timeZone = new DateTimeZone('+0100');

документы:

http://php.net/manual/en/datetimezone.construct.php

примечание: по этой ссылке документации это новое использование конструктора было доступно с версии PHP 5.5.10.


эта работа для меня:

function Get_Offset_Date($Offset, $Unixtime = false, $Date_Format = 'Y.m.d,H:i')
{
    try
    {
        $date   = new DateTime("now",new DateTimeZone($Offset));
        if($Unixtime)
        {
            $date->setTimestamp($Unixtime);
        }

        $Unixtime = $date->getTimestamp();
        return $date->format($Date_Format);
    }

    catch(Exception $e)
    {
        return false;
    }
}

чтобы использовать его, просто вызовите это, чтобы получить текущее время:

echo Get_Offset_Date('+2:00');

и это для даты смещения get (вход-unixtime)

echo Get_Offset_Date('+2:00',1524783867);