C# 2.0 определенный часовой пояс
Я работаю над унаследованным приложением в.NET 2.0 и хочу преобразовать время из местного времени (например, UTC+1) в время в Бразилии (то есть, по стандарту Windows, стандартное время E. South America) и назад.
Я собрал этот код, который придумал для преобразования:
using System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;
using Microsoft.Win32;
namespace timezone
{
class Program
{
[DllImport("kernel32.dll", CharSet = CharSet.Auto, ExactSpelling = false)]
private static extern int SystemTimeToTzSpecificLocalTime(ref
TIME_ZONE_INFORMATION lpTimeZone, ref SYSTEMTIME lpUniversalTIme, out
SYSTEMTIME lpLocalTime);
[DllImport("kernel32.dll", CharSet = CharSet.Auto, ExactSpelling = false)]
private static extern int TzSpecificLocalTimeToSystemTime(ref
TIME_ZONE_INFORMATION lpTimeZone, ref SYSTEMTIME lpLocalTime, out SYSTEMTIME
lpUniversalTIme);
[DllImport("kernel32.dll")]
private static extern void GetSystemTime(out SYSTEMTIME lpSystemTime);
[StructLayout(LayoutKind.Sequential)]
private struct SYSTEMTIME
{
public ushort wYear;
public ushort wMonth;
public ushort wDayOfWeek;
public ushort wDay;
public ushort wHour;
public ushort wMinute;
public ushort wSecond;
public ushort wMilliseconds;
}
//Registry time zone format. See KB article Q115231
[StructLayout(LayoutKind.Sequential)]
private struct REG_TIME_ZONE_INFORMATION
{
public int Bias;
public int StandardBias;
public int DaylightBias;
public SYSTEMTIME StandardDate;
public SYSTEMTIME DaylightDate;
}
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
private struct TIME_ZONE_INFORMATION
{
public int Bias;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 32)]
public string StandardName;
public SYSTEMTIME StandardDate;
public int StandardBias;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 32)]
public string DaylightName;
public SYSTEMTIME DaylightDate;
public int DaylightBias;
}
private static List<TIME_ZONE_INFORMATION> GetTimeZones()
{
List<TIME_ZONE_INFORMATION> list = new List<TIME_ZONE_INFORMATION>();
RegistryKey key =
Registry.LocalMachine.OpenSubKey(@"SOFTWARE\Microsoft\Windows NT\CurrentVersion\Time Zones");
if (key == null)
return list;
string[] subKeyNames = key.GetSubKeyNames();
foreach (string subKeyName in subKeyNames)
{
RegistryKey subKey = key.OpenSubKey(subKeyName);
if (subKey != null)
{
object value = subKey.GetValue("TZI");
if (value != null)
{
int length =
Marshal.SizeOf(typeof(REG_TIME_ZONE_INFORMATION));
IntPtr p = Marshal.AllocHGlobal(length);
Marshal.Copy((byte[])value, 0, p, length);
REG_TIME_ZONE_INFORMATION rtzi =
(REG_TIME_ZONE_INFORMATION)Marshal.PtrToStructure(p,
typeof(REG_TIME_ZONE_INFORMATION));
Marshal.FreeHGlobal(p);
TIME_ZONE_INFORMATION tzi = new TIME_ZONE_INFORMATION();
tzi.Bias = rtzi.Bias;
tzi.DaylightBias = rtzi.DaylightBias;
tzi.StandardBias = rtzi.StandardBias;
tzi.DaylightDate = rtzi.DaylightDate;
tzi.StandardDate = rtzi.StandardDate;
tzi.DaylightName = (string)subKey.GetValue("Dlt", "");
tzi.StandardName = (string)subKey.GetValue("Std", "");
list.Add(tzi);
}
subKey.Close();
}
}
key.Close();
return list;
}
static void Main(string[] args)
{
foreach (TIME_ZONE_INFORMATION tzi in GetTimeZones())
{
if ("E. South America Standard Time" == tzi.StandardName)
{
Console.WriteLine("local time: " + DateTime.Now);
Console.WriteLine("name: {0} st bias:{1} daylight date:{2}-{3}-{4}, bias:{5}", tzi.StandardName, tzi.DaylightBias, tzi.DaylightDate.wDay, tzi.DaylightDate.wMonth, tzi.DaylightDate.wYear, tzi.Bias);
TIME_ZONE_INFORMATION tzi2 = tzi; //local copy so that i can pass it as a ref
SYSTEMTIME braziltime = new SYSTEMTIME();
SYSTEMTIME localtime = new SYSTEMTIME();
GetSystemTime(out localtime);
SystemTimeToTzSpecificLocalTime(ref tzi2, ref localtime, out braziltime);
Console.WriteLine("{0}:{1}", braziltime.wHour, braziltime.wMinute);
Console.WriteLine("{0}-{1}-{2} {3}:{4}:{5}", braziltime.wYear, braziltime.wMonth, braziltime.wDay, braziltime.wHour, braziltime.wMinute, braziltime.wSecond);
DateTime dt = DateTime.Now;
braziltime.wYear = (ushort)dt.Year;
braziltime.wMonth = (ushort)dt.Month;
braziltime.wDay = (ushort)dt.Day;
braziltime.wHour = (ushort)(dt.Hour - 3); //today the timezone difference is 3 hours
braziltime.wMinute = (ushort)dt.Minute;
braziltime.wSecond = (ushort)dt.Second;
TzSpecificLocalTimeToSystemTime(ref tzi2, ref braziltime, out localtime);
Console.WriteLine("{0}-{1}-{2} {3}:{4}:{5}", localtime.wYear, localtime.wMonth, localtime.wDay, localtime.wHour, localtime.wMinute, localtime.wSecond);
break;
}
}
Console.ReadLine();
}
}
}
но я получаю этот вывод:
local time: 11/22/2010 11:55:15 AM
name: E. South America Standard Time st bias:-60 daylight date:3-10-0, bias:180 8:55
2010-11-22 8:55:15
2010-11-22 10:55:15
Поэтому я беру местное время и перевожу его в бразильское время и обратно и получаю на час меньше. Есть идеи что не так?
3 ответа
Я думаю, вы ожидаете, что первая строка вашего вывода будет соответствовать последней строке. Этого не происходит, потому что первая строка записывает местное время. Затем вы вызываете GetSystemTime и конвертируете это значение в Бразилию снова и снова. GetSystemTime возвращает UTC, поэтому значение, которое вы возвращаете, а затем выводите в последней строке, также должно быть UTC. Другими словами, вы не сравниваете подобное с подобным.
Если вы выводите значение localtime сразу после вызова GetSystemTime, вы должны увидеть, что оно соответствует выходному значению после преобразования.
Если вы хотите перевести ваше местное время в бразильское, то вам, вероятно, нужно преобразовать ваше местное время в UTC, а затем преобразовать UTC в бразильское время, используя соответствующую информацию о часовом поясе для каждого шага.
Помните, что DateTime - это просто структура для хранения даты и времени.
Вы должны использовать везде в своем приложении UTC, и просто использовать Locale для вывода. Чтобы быть чистым, я предпочитаю использовать большую часть времени utc и делать конвертирование из него.
Если бы это было 3,5...
using System;
// ReSharper disable once CheckNamespace
public static class BrazilTime
{
public static DateTime Now
{
get
{
return TimeZoneInfo.ConvertTimeFromUtc(DateTime.UtcNow, TimeZoneInfo.FindSystemTimeZoneById("E. South America Standard Time"));
}
}
}