Как вы получаете информацию для произвольного часового пояса в Windows?

В идеале я хотел бы иметь возможность взять имя часового пояса и запросить у Windows информацию о соответствующем часовом поясе (смещение по UTC, смещение летнего времени, даты переключения DST и т. Д.). Похоже, что Windows использует структуру TIME_ZONE_INFORMATION для хранения такого рода информации. Итак, предположительно, я хочу функцию, которая принимает строку с именем часового пояса и возвращает структуру TIME_ZONE_INFORMATION.

Однако все, что я могу найти, это функции, такие как GetTimeZoneInformation(), которые дают мне TIME_ZONE_INFORMATION для местного времени. Что мне нужно, это функция, которая даст мне эту информацию для произвольного часового пояса, независимо от того, что местный часовой пояс.

Единственный способ получить эту информацию - получить ее напрямую из реестра, что далеко не идеально. Страница TIME_ZONE_INFORMATION показывает, где она находится в реестре, поэтому должна быть возможность получить информацию оттуда, но я бы предпочел, чтобы для этого была правильная системная функция. Существует ли такая функция, или мне нужно заняться регистрацией, чтобы получить информацию о часовом поясе для произвольного часового пояса?

2 ответа

Решение

Информация о часовом поясе содержится в виде двоичных данных в реестре в HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Time Zones\(zone name)\TZI, Структура данных приведена в документации TIME_ZONE_INFORMATION:

struct STimeZoneFromRegistry
{
 long  Bias;
 long  StandardBias;
 long  DaylightBias;
 SYSTEMTIME StandardDate;
 SYSTEMTIME DaylightDate;
};

А вот пример кода для чтения ключа:

TIME_ZONE_INFORMATION tz = {0};
STimeZoneFromRegistry binary_data;
DWORD size = sizeof(binary_data);
HKEY hk = NULL;
TCHAR zone_key[] = _T("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Time Zones\\Central Standard Time");
if ((RegOpenKeyEx(HKEY_LOCAL_MACHINE, zone_key, 0, KEY_QUERY_VALUE, &hk) == ERROR_SUCCESS) &&
 (RegQueryValueEx(hk, "TZI", NULL, NULL, (BYTE *) &binary_data, &size) == ERROR_SUCCESS))
{
 tz.Bias = binary_data.Bias;
 tz.DaylightBias = binary_data.DaylightBias;
 tz.DaylightDate = binary_data.DaylightDate;
 tz.StandardBias = binary_data.StandardBias;
 tz.StandardDate = binary_data.StandardDate;
}

Изменить: Извините, этот ответ является избыточным - я уверен, что вы могли бы все это выяснить, используя документацию, с которой вы связаны в вопросе. Я должен был сделать это только один раз, и это единственный метод, который я смог найти.

Вот пример без реестра. Информация о часовом поясе зависит от года, поэтому я добавил параметр для ее хранения.

TIME_ZONE_INFORMATION GmtTimezone( const wstring& name, SYSTEMTIME utc )
{
    DYNAMIC_TIME_ZONE_INFORMATION dynamicTimezone = {};
    DWORD result=0;
    for( DWORD i = 0; result!=ERROR_NO_MORE_ITEMS; ++i )
    {
        result = ::EnumDynamicTimeZoneInformation( i, &dynamicTimezone );
        if( result==ERROR_SUCCESS && name==dynamicTimezone.StandardName )
            break;
    }
    if( result!=ERROR_SUCCESS )
        throw L"Could not find timezone "+name;

    TIME_ZONE_INFORMATION tz;
    if( !GetTimeZoneInformationForYear(static_cast<USHORT>(utc.wYear), &dynamicTimezone, &tz) )
        throw "GetTimeZoneInformationForYear failed"+std::to_string(GetLastError());
    return tz;
    //SYSTEMTIME localTime;
    //SystemTimeToTzSpecificLocalTime( &tz, &utc, &localTime );
    //auto offsetHours = localTime.wHour-utc.wHour;
}
Другие вопросы по тегам