繁体   English   中英

C#在特定的TimeZone中添加1天到DateTimeOffset

[英]C# Add 1 day in specific TimeZone to DateTimeOffset

我有一个DateTimeOffset实例,并且需要考虑夏时制规则(因此可能导致Offset更改)在特定的TimeZone (西欧标准时间)中添加1天。 没有第三方库怎么办?

可验证的示例:

using System;
using Microsoft.VisualStudio.TestTools.UnitTesting;

namespace UnitTestProject
{
    [TestClass]
    public class TimeZoneTests
    {
        [TestMethod]
        public void DateTimeOffsetAddDays_DaylightSaving_OffsetChange()
        {
            var timeZoneId = "W. Europe Standard Time";
            var utcTimestamp = new DateTimeOffset(2017, 10, 28, 22, 0, 0, TimeZoneInfo.Utc.BaseUtcOffset);
            var weuropeStandardTimeTimestamp = TimeZoneInfo.ConvertTimeBySystemTimeZoneId(utcTimestamp, timeZoneId);
            Assert.AreEqual(new DateTime(2017, 10, 29), weuropeStandardTimeTimestamp.DateTime);
            Assert.AreEqual(TimeSpan.FromHours(2), weuropeStandardTimeTimestamp.Offset);

            var weuropeStandardTimeTimestampNextDay = AddDaysInTimeZone(weuropeStandardTimeTimestamp, 1, timeZoneId);

            Assert.AreEqual(new DateTime(2017, 10, 30), weuropeStandardTimeTimestampNextDay);
            Assert.AreEqual(TimeSpan.FromHours(1), weuropeStandardTimeTimestamp.Offset);
        }

        private DateTimeOffset AddDaysInTimeZone(DateTimeOffset timestamp, int days, string timeZoneId)
        {
            // this line has to be fixed:
            return timestamp.AddDays(days);
        }
    }
}

AddDaysInTimeZone方法应替换为正确的实现。

附言:如果导致无效/模糊/跳过日期,则可以抛出异常。

TimeZoneInfo使此操作相当简单-只需在值的DateTime部分添加一天,检查结果是否被跳过或模棱两可,如果不是,则向该区域询问UTC偏移量。 这是一个完整的示例,显示了所有不同的可能性:

using System;
using System.Globalization;

using static System.FormattableString;

class Program
{
    static void Main()
    {        
        // Stay in winter
        Test("2017-01-22T15:00:00+01:00");
        // Skipped time during transition
        Test("2017-03-25T02:30:00+01:00");
        // Offset change to summer
        Test("2017-03-25T15:00:00+01:00");
        // Stay in summer
        Test("2017-06-22T15:00:00+02:00");
        // Ambiguous time during transition
        Test("2017-10-28T02:30:00+02:00");
        // Offset change back to winter
        Test("2017-10-28T15:00:00+02:00");
        // Stay in winter
        Test("2017-12-22T15:00:00+01:00");
    }

    static void Test(string startText)
    {
        var zone = TimeZoneInfo.FindSystemTimeZoneById("W. Europe Standard Time");
        var start = DateTimeOffset.ParseExact(
            startText, "yyyy-MM-dd'T'HH:mm:ssK", CultureInfo.InvariantCulture);
        try
        {
            var end = AddOneDay(start, zone);
            Console.WriteLine(Invariant($"{startText} => {end:yyyy-MM-dd'T'HH:mm:ssK}"));
        }
        catch (Exception e)
        {
            Console.WriteLine($"{startText} => {e.Message}");
        }
    }

    static DateTimeOffset AddOneDay(DateTimeOffset start, TimeZoneInfo zone)
    {
        var newLocal = start.DateTime.AddDays(1);
        // TODO: Use a better exception type :)
        if (zone.IsAmbiguousTime(newLocal))
        {
            throw new Exception("Ambiguous");
        }
        if (zone.IsInvalidTime(newLocal))
        {
            throw new Exception("Skipped");
        }
        return new DateTimeOffset(newLocal, zone.GetUtcOffset(newLocal));
    }

}

暂无
暂无

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

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