如何从 NodaTime 中的偏移量获取所有 IANA 时区?

How to get all IANA Timezones from an offset in NodaTime?

在我们的 ASP.NET MVC 5 应用程序中,我们有带有时区 ID (IANA) 的用户配置文件。

public class User
{
    public int Id { get; set; }
    public string TimeZoneId { get; set; }
}

任务是在当地时间早上 6 点向所有用户发送电子邮件。我们的计划是每小时向本地时间为早上 6 点(根据服务器的 UTC 时间计算)的所有时区发送电子邮件。

由于我们只有时区 ID,我希望获得相应的时区 ID,比方说,偏移量 -4:00:00 -- 考虑到 DST.

string[] timezoneIds;

然后我可以像这样查询我的数据库:

db.Users.Where(x => timezoneIds.Contains(x.TimeZoneId));

显然,我的问题是,这是一个不错的想法,还是有我不知道的针对此问题的最佳实践?

谢谢。

您可以尝试以下方法:

首先创建一个包含所有偏移量和属于该偏移量的时区的字典

Instant now = SystemClock.Instance.Now;
IDateTimeZoneProvider timeZoneProvider = DateTimeZoneProviders.Tzdb;
Dictionary<TimeSpan, List<string>> timezonesForOffset = new Dictionary<TimeSpan, List<string>>();
foreach(var id in timeZoneProvider.Ids){
    ZonedDateTime zdt = now.InZone(timeZoneProvider[id]);
    var timespan = zdt.Offset.ToTimeSpan();
    if(timezonesForOffset.ContainsKey(timespan)){
        timezonesForOffset[timespan].Add(id);
    } else {
        timezonesForOffset[timespan] = new List<string> {id, };
    }
}

完成后,您可以使用以下代码段获取特定时区内的所有用户

var timespan2 = new TimeSpan(1,0,0);
var timezonesWithOffset = timezonesForOffset[timespan2];

var usersinTimezone = db.Users.Where(x=> timezonesWithOffset.Contains(x.TimezoneId)).ToList();

这将获得时区 utc+1 中的所有用户

Serge 展示的“构建字典”的更简单代码:使用 LINQ 的查找、DateTimeZone.GetUtcOffsetOffset 键而不是 TimeSpan 值:

var now = SystemClock.Instance.GetCurrentInstant();
var provider = DateTimeZoneProviders.Tzdb;
var idsByCurrentOffset = provider.Ids.ToLookup(id => provider[id].GetUtcOffset(now));

那么你可以使用:

var offset = Offset.FromHours(5);
foreach (var id in idsByCurrentOffset[offset])
{
    ...
}

(您甚至不需要先检查是否存在,因为查找索引器 returns 当出现查找中不存在的键时是一个空序列。)