繁体   English   中英

C#2.0特定时区

[英]C# 2.0 specific timezone

我正在.NET 2.0中的旧版应用程序上工作,我想将时间从本地时间(恰好是UTC + 1)转换为巴西的时间(Windows称之为“ E.南美标准时间”)。然后回来。

我将以下代码整合在一起:

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

因此,我将当地时间转换为巴西时间,然后减少了一个小时。 任何想法有什么问题吗?

我认为您期望输出的第一行与最后一行匹配。 不会发生这种情况,因为第一行写的是本地时间。 然后,您调用GetSystemTime并将该值转换为巴西时间,然后再次返回。 GetSystemTime返回UTC,所以您返回并在最后一行输出的值也应为UTC。 换句话说,您并没有将“喜欢”与“喜欢”进行比较。

如果在调用GetSystemTime之后立即输出localtime的值,则应该看到该值与转换后输出的值匹配。

如果要将本地时间转换为巴西时间,则可能需要将本地时间转换为UTC,然后使用适合每个步骤的时区信息将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"));
        }
    }
}

请记住,DateTime只是为了存储datetime的结构。

您应该在应用UTC中使用任何地方,而只使用Locale进行输出。 为了保持清洁,我更喜欢使用utc的大部分时间并从中进行转换。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM