我正在处理我最可怕的噩梦——时区和夏令时。我已经读了很多关于stackoverflow的帖子,但我仍然无法理解。问题是:我正在向API请求一天的UTC数据,但我所使用的系统需要在美国/太平洋时间请求。文件上写着:
-
报告范围筛选支持时区,但所有响应都在美国太平洋时区返回,请相应调整夏令时结束。
-
DST开始后的API调用应附加-07:00,DST结束后的调用应附加-0800
-
2018夏令时从2018年3月11日星期日凌晨2点开始。
-
3月11日之前的API调用应具有以下语法:
&start=2017-03-10T00:00:00-08:00&end=2017-03-10T23:59:59-08:00
-
实际夏令时的API调用应具有以下语法:
&start=2018-03-11T00:00:00-08:00&end=2017-03-11T23:59:59-07:00
-
除了令人困惑的2017年和2018年的混合之外,没有实际的参数来指定您需要的时区,但您必须调整以下格式的数据:2018-03-11T00:00:00-08:00
。
对我来说,它看起来像ISO格式,但我花了很长时间试图获得yyyy-MM-dd'T'HH:mm:ssXXX
而不是'yyyy-MM-dd'T'HH:mm:ss.SSSXXX'
,但无法实现。所以我创建了以下解决方法:
def dst_calc(single_date):
zone = pytz.timezone("US/Pacific")
day = single_date.strftime("%Y-%m-%d")
tdelta_1 = datetime.strptime('2:00:00', '%H:%M:%S') - datetime.strptime('1:00:00', '%H:%M:%S')
tdelta_0 = datetime.strptime('1:00:00', '%H:%M:%S') - datetime.strptime('1:00:00', '%H:%M:%S')
logger.info('check for DST')
if zone.localize(datetime(single_date.year, single_date.month, single_date.day)).dst() == tdelta_1:
logger.info('summertime')
start = single_date.strftime("%Y-%m-%d") + "T00:00:00-07:00"
end = single_date.strftime("%Y-%m-%d") + "T23:59:59-07:00"
elif zone.localize(datetime(single_date.year, single_date.month, single_date.day) + timedelta(days=1)).dst() == tdelta_1:
logger.info('beginning of summertime')
start = single_date.strftime("%Y-%m-%d") + "T00:00:00-08:00"
end = single_date.strftime("%Y-%m-%d") + "T23:59:59-07:00"
elif zone.localize(datetime(single_date.year, single_date.month, single_date.day)).dst() == tdelta_0:
logger.info('wintertime')
start = single_date.strftime("%Y-%m-%d") + "T00:00:00-08:00"
end = single_date.strftime("%Y-%m-%d") + "T23:59:59-08:00"
显然,这只在美国/太平洋时区,为了得到UTC日期,我需要从开始时间和8个时间戳中减去8小时的差值,即T16:00:00-08:00
,但我想知道是否有更好的方法/包/格式化程序可以做到这一点,这是一种更符合逻辑的方法。
您可以使用datetime的astimezone
方法来确定正确的小时数。
import datetime, pytz
now = datetime.datetime.now() # datetime.datetime(2019, 2, 12, 17, 0, 0, 0)
now.astimezone(pytz.utc)
# datetime.datetime(2019, 2, 12, 16, 0, 0, 0, tzinfo=<UTC>)
now.astimezone(pytz.timezone('US/Pacific'))
# datetime.datetime(2019, 2, 12, 8, 0, 0, 0, tzinfo=<DstTzInfo 'US/Pacific' PST-1 day, 16:00:00 STD>)