使用Python查找时区当前是否处于夏令时状态

28
我们有一台服务器运行在GMT时间。我需要编写一个Python脚本,确定洛杉矶现在(此时此刻)是否为夏令时。我查看了pytz和time,但是无法弄清楚。我意识到我可以创建一些逻辑比较LA的当前时间和GMT时间,但如果我能使用标准库,它会更加简洁。
谢谢。
编辑:这里是我设置时区的一些示例代码:
from pytz import timezone
import pytz
from datetime import datetime, timedelta

tz = timezone('America/Los_Angeles')
// Instantiate a datetime object using tz? 

编辑:这里是一个可以使用的代码片段。它不够简洁,这也是我在问是否有专门用于此类问题的库或函数,比如 is_dst() 函数。

utc = timezone("UTC")
now = utc.localize(datetime.utcnow())
los_angeles_tz = timezone('America/Los_Angeles')
los_angeles_time = now.astimezone(los_angeles_tz)

delta = los_angeles_time.utcoffset()
dstDelta = timedelta(hours=-8)

is_dst = (delta == dstDelta)

1
https://dev59.com/mmsz5IYBdhLWcg3wTl-r - kgu87
4
公平地说,这并不是Python库中最友好的领域,尝试解决问题有一定难度。因此,虽然严格意义上来说不属于该主题,但提出这个问题也并不算不合理。 - Jon Clements
2
@JonClements 也许是这样,但看到一些证明 OP 超越了打开文档并说“哇这看起来很难,也许 SO 上的某个人可以解决我的问题”的东西会很好。 - Marcin
相关:Python 夏令时 - jfs
2个回答

43
import pytz
from datetime import datetime, timedelta

def is_dst(zonename):
    tz = pytz.timezone(zonename)
    now = pytz.utc.localize(datetime.utcnow())
    return now.astimezone(tz).dst() != timedelta(0)

用法:

>>> is_dst("America/Los_Angeles")
False

>>> is_dst("America/Sao_Paulo")
True

1
你可以使用 datetime.now(tz) 获取以 tz 时区为基准的当前时间,例如:is_dst = lambda zonename: bool(datetime.now(pytz.timezone(zonename)).dst()) - jfs
3
@J.F.Sebastian - pytz文档明确指出需要使用localize函数,将pytz时区传递给datetime的tzinfo参数是不够的。这是否错误? - Matt Johnson-Pint
1
现在(tz)通常情况下和特别指定UTC时区是错误的。now()文档中调用tz.fromutc()被pytz时区重载以执行正确操作(使用_utc_transition_times获取有效的tzinfo)。此外,UTC是一个特殊情况:零UTC偏移,始终如一。 - jfs

12
from datetime import datetime
import pytz

isdst_now_in = lambda zonename: bool(datetime.now(pytz.timezone(zonename)).dst())

例子:

>>> isdst_now_in("America/Los_Angeles") # 2014-10-27 12:32:07 PDT-0700
True
>>> isdst_now_in("Australia/Melbourne") # 2014-10-28 06:32:07 AEDT+1100
True

这是Unix上仅使用time.tzset()函数的标准库版本:

import os
import time
from contextlib import contextmanager

@contextmanager
def local_timezone(zonename):
    #NOTE: it manipulates global state
    oldname = os.environ.get('TZ')
    try:
        os.environ['TZ'] = zonename
        time.tzset()
        yield
    finally:
        if oldname is None:
           del os.environ['TZ']
        else:
           os.environ['TZ'] = oldname
        time.tzset()

def time_isdst_now_in(zonename):
    with local_timezone(zonename):
         return time.localtime().tm_isdst > 0

使用方法相同:

>>> time_isdst_now_in('Europe/London') # 2014-04-17 18:42:11 BST+0100
True

注意:由于某些边缘情况的问题time.daylight未被使用。


Note: time.daylight因为一些边缘情况的问题而未被使用。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接