我正在尝试创建一个单元测试,以测试当机器上的时区因设置不正确而更改时的情况。

在测试中,我需要能够在非本地时区创建DateTime对象,以确保运行测试的人可以成功地这样做,而不管他们位于哪里。

从我可以从DateTime构造函数中看到,我可以将TimeZone设置为本地时区,UTC时区或未指定。

我如何创建一个日期时间与特定的时区,如PST?


当前回答

使用TimeZones类可以很容易地创建时区特定的日期。

TimeZoneInfo.ConvertTime(DateTime.Now, TimeZoneInfo.FindSystemTimeZoneById(TimeZones.Paris.Id));

其他回答

这里的其他答案是有用的,但他们不包括如何访问太平洋具体-在这里:

public static DateTime GmtToPacific(DateTime dateTime)
{
    return TimeZoneInfo.ConvertTimeFromUtc(dateTime,
        TimeZoneInfo.FindSystemTimeZoneById("Pacific Standard Time"));
}

奇怪的是,虽然“太平洋标准时间”通常与“太平洋夏令时”的意思不同,但在这种情况下,它指的是一般的太平洋时间。事实上,如果您使用FindSystemTimeZoneById来获取它,其中一个可用的属性是bool,告诉您该时区当前是否处于夏令时。

你可以在一个库中看到更通用的例子,我最终拼凑了一个库来处理我需要的不同时区的DateTimes,基于用户从哪里询问,等等:

https://github.com/b9chris/TimeZoneInfoLib.Net

这将不会在Windows之外工作(例如Linux上的Mono),因为时间列表来自Windows注册表: HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Time Zones\

在下面你会发现键(注册表编辑器中的文件夹图标);这些键的名字是你传递给FindSystemTimeZoneById的。在Linux上,您必须使用单独的Linux标准时区定义集,对此我还没有充分探讨。

您必须为此创建一个自定义对象。您的自定义对象将包含两个值:

DateTime值 一个时区对象

不确定是否已经有clr提供的数据类型具有此功能,但至少TimeZone组件已经可用。

使用TimeZones类可以很容易地创建时区特定的日期。

TimeZoneInfo.ConvertTime(DateTime.Now, TimeZoneInfo.FindSystemTimeZoneById(TimeZones.Paris.Id));

Jon的回答谈到了TimeZone,但我建议使用TimeZoneInfo代替。

就我个人而言,我喜欢尽可能将内容保存在UTC中(至少对于过去;为未来存储UTC有潜在的问题),所以我建议一个这样的结构:

public struct DateTimeWithZone
{
    private readonly DateTime utcDateTime;
    private readonly TimeZoneInfo timeZone;

    public DateTimeWithZone(DateTime dateTime, TimeZoneInfo timeZone)
    {
        var dateTimeUnspec = DateTime.SpecifyKind(dateTime, DateTimeKind.Unspecified);
        utcDateTime = TimeZoneInfo.ConvertTimeToUtc(dateTimeUnspec, timeZone); 
        this.timeZone = timeZone;
    }

    public DateTime UniversalTime { get { return utcDateTime; } }

    public TimeZoneInfo TimeZone { get { return timeZone; } }

    public DateTime LocalTime
    { 
        get 
        { 
            return TimeZoneInfo.ConvertTime(utcDateTime, timeZone); 
        }
    }        
}

您可能希望将“TimeZone”名称更改为“TimeZoneInfo”以使事情更清楚-我自己更喜欢简短的名称。

我喜欢乔恩·斯基特的回答,但我想补充一点。我不确定乔恩是否期待着ctor总是在本地时区通过。但我想把它用在本地以外的情况下。

我正在从数据库中读取值,并且我知道该数据库位于哪个时区。在ctor中,我将传入数据库的时区。但是我想知道当地时间的值。Jon的LocalTime不会返回转换为本地时区日期的原始日期。它返回转换成原始时区的日期(无论您传递给ctor的是什么)。

我认为这些属性名称清楚了…

public DateTime TimeInOriginalZone { get { return TimeZoneInfo.ConvertTime(utcDateTime, timeZone); } }
public DateTime TimeInLocalZone    { get { return TimeZoneInfo.ConvertTime(utcDateTime, TimeZoneInfo.Local); } }
public DateTime TimeInSpecificZone(TimeZoneInfo tz)
{
    return TimeZoneInfo.ConvertTime(utcDateTime, tz);
}