Разбор упорядоченных временных меток по местному времени (к UTC) при соблюдении летнего времени
У меня есть файлы данных CSV с записями с метками времени по местному времени. К сожалению, файлы данных охватывают период, когда изменяется летнее время (3 ноября 2013 г.), поэтому временная составляющая временных меток для записей идет: 12:45, 1:00, 1:15, 1:30, 1:45, 1:00, 1:15, 1:30, 1:45, 2:00
, Я хочу иметь возможность конвертировать и хранить значения в базе данных как UTC.
К сожалению стандарт DateTime.Parse()
Функция.NET будет разбираться так (все 3 ноября 2013 г.):
| Time String | Parsed Local Time | In DST | Parsed Local Time to UTC
| 12:45 am | 12:45 am | Yes | 4:45 am
| 12:59:59 am | 12:59:59 am | Yes | 4:59:59 am
| 01:00 am | 1:00 am | No | 6:00 am
| 01:15 am | 1:15 am | No | 6:15 am
Так что он никогда не видит 1:00-1:59:59 am
диапазон, как в DST, и мои проанализированные метки времени в UTC скачков в час.
Есть ли библиотека или класс, который позволит мне анализировать метки времени и учитывать изменения в летнем времени? Как какой-то инстанцируемый класс, который будет помнить поток временных меток, которые он уже получил, и соответствующим образом корректировать проанализированный временной штамп?
Предположения о данных, которые могут быть сделаны при разборе:
- У меня есть время начала файла (отметка времени первой записи) в разделе заголовка файла в локальном и UTC.
- Записи в порядке по метке времени
- Все местное время в восточном стандарте
- Данные могут также пойти другим путем: из DST в это
- Записи содержат полную метку времени в формате:
yyyy/mm/dd HH:mm:ss
(2013/11/03 00:45:00
)
Примечание. В то время как мое программное обеспечение находится на C#, я специально не помечал C#/.NET, поскольку полагал, что могу использовать реализацию решения на любом языке и при необходимости перекодировать.
2 ответа
В C#:
// Define the input values.
string[] input =
{
"2013-11-03 00:45:00",
"2013-11-03 01:00:00",
"2013-11-03 01:15:00",
"2013-11-03 01:30:00",
"2013-11-03 01:45:00",
"2013-11-03 01:00:00",
"2013-11-03 01:15:00",
"2013-11-03 01:30:00",
"2013-11-03 01:45:00",
"2013-11-03 02:00:00",
};
// Get the time zone the input is meant to be interpreted in.
TimeZoneInfo tz = TimeZoneInfo.FindSystemTimeZoneById("Eastern Standard Time");
// Create an array for the output values
DateTimeOffset[] output = new DateTimeOffset[input.Length];
// Start with the assumption that DST is active, as ambiguities occur when moving
// out of daylight time into standard time.
bool dst = true;
// Iterate through the input.
for (int i = 0; i < input.Length; i++)
{
// Parse the input string as a DateTime with Unspecified kind
DateTime dt = DateTime.ParseExact(input[i], "yyyy-MM-dd HH:mm:ss",
CultureInfo.InvariantCulture);
// Determine the offset.
TimeSpan offset;
if (tz.IsAmbiguousTime(dt))
{
// Get the possible offsets, and use the DST flag and the previous entry
// to determine if we are past the transition point. This only works
// because we have outside knowledge that the items are in sequence.
TimeSpan[] offsets = tz.GetAmbiguousTimeOffsets(dt);
offset = dst && (i == 0 || dt >= output[i - 1].DateTime)
? offsets[1] : offsets[0];
}
else
{
// The value is unambiguous, so just get the single offset it can be.
offset = tz.GetUtcOffset(dt);
}
// Use the determined values to construct a DateTimeOffset
DateTimeOffset dto = new DateTimeOffset(dt, offset);
// We can unambiguously check a DateTimeOffset for daylight saving time,
// which sets up the DST flag for the next iteration.
dst = tz.IsDaylightSavingTime(dto);
// Save the DateTimeOffset to the output array.
output[i] = dto;
}
// Show the output for debugging
foreach (var dto in output)
{
Console.WriteLine("{0:yyyy-MM-dd HH:mm:ss zzzz} => {1:yyyy-MM-dd HH:mm:ss} UTC",
dto, dto.UtcDateTime);
}
Выход:
2013-11-03 00:45:00 -04:00 => 2013-11-03 04:45:00 UTC
2013-11-03 01:00:00 -04:00 => 2013-11-03 05:00:00 UTC
2013-11-03 01:15:00 -04:00 => 2013-11-03 05:15:00 UTC
2013-11-03 01:30:00 -04:00 => 2013-11-03 05:30:00 UTC
2013-11-03 01:45:00 -04:00 => 2013-11-03 05:45:00 UTC
2013-11-03 01:00:00 -05:00 => 2013-11-03 06:00:00 UTC
2013-11-03 01:15:00 -05:00 => 2013-11-03 06:15:00 UTC
2013-11-03 01:30:00 -05:00 => 2013-11-03 06:30:00 UTC
2013-11-03 01:45:00 -05:00 => 2013-11-03 06:45:00 UTC
2013-11-03 02:00:00 -05:00 => 2013-11-03 07:00:00 UTC
Обратите внимание, что это предполагает, что в первый раз вы встретите неоднозначное время, такое как 1:00, когда оно будет в летнее время. Скажем, ваш список был усечен до последних 5 записей - вы не знали бы, что они были в стандартное время. В этом конкретном случае вы мало что можете сделать.
Если последовательные временные метки не могут идти в обратном направлении, если они выражены как время в UTC, тогда этот скрипт Python может преобразовать местное время в UTC:
#!/usr/bin/env python3
import sys
from datetime import datetime, timedelta
import pytz # $ pip install pytz
tz = pytz.timezone('America/New_York' if len(sys.argv) < 2 else sys.argv[1])
previous = None #XXX set it from UTC time: `first_entry_utc.astimezone(tz)`
for line in sys.stdin: # read from stdin
naive = datetime.strptime(line.strip(), "%Y/%m/%d %H:%M:%S") # no timezone
try:
local = tz.localize(naive, is_dst=None) # attach timezone info
except pytz.AmbiguousTimeError:
# assume ambiguous time always corresponds to True -> False transition
local = tz.localize(naive, is_dst=True)
if previous >= local: # timestamps must be increasing
local = tz.localize(naive, is_dst=False)
assert previous < local
#NOTE: allow NonExistentTimeError to propagate (there shouldn't be
# invalid local times in the input)
previous = local
utc = local.astimezone(pytz.utc)
timestamp = utc.timestamp()
time_format = "%Y-%m-%d %H:%M:%S %Z%z"
print("{local:{time_format}}; {utc:{time_format}}; {timestamp:.0f}"
.format_map(vars()))
вход
2013/11/03 00:45:00
2013/11/03 01:00:00
2013/11/03 01:15:00
2013/11/03 01:30:00
2013/11/03 01:45:00
2013/11/03 01:00:00
2013/11/03 01:15:00
2013/11/03 01:30:00
2013/11/03 01:45:00
2013/11/03 02:00:00
Выход
2013-11-03 00:45:00 EDT-0400; 2013-11-03 04:45:00 UTC+0000; 1383453900
2013-11-03 01:00:00 EDT-0400; 2013-11-03 05:00:00 UTC+0000; 1383454800
2013-11-03 01:15:00 EDT-0400; 2013-11-03 05:15:00 UTC+0000; 1383455700
2013-11-03 01:30:00 EDT-0400; 2013-11-03 05:30:00 UTC+0000; 1383456600
2013-11-03 01:45:00 EDT-0400; 2013-11-03 05:45:00 UTC+0000; 1383457500
2013-11-03 01:00:00 EST-0500; 2013-11-03 06:00:00 UTC+0000; 1383458400
2013-11-03 01:15:00 EST-0500; 2013-11-03 06:15:00 UTC+0000; 1383459300
2013-11-03 01:30:00 EST-0500; 2013-11-03 06:30:00 UTC+0000; 1383460200
2013-11-03 01:45:00 EST-0500; 2013-11-03 06:45:00 UTC+0000; 1383461100
2013-11-03 02:00:00 EST-0500; 2013-11-03 07:00:00 UTC+0000; 1383462000