如何将本地时间的datetime字符串转换为UTC时间的字符串?

我确信我以前做过这个,但找不到它,所以希望将来能帮助我(和其他人)做到这一点。

澄清:例如,如果我的本地时区(+10)是2008-09-17 14:02:00,我希望生成一个具有等效UTC时间的字符串:2008-09-17 04:02:00。

此外,从http://lucumr.pocoo.org/2011/7/15/eppur-si-muove/请注意,一般来说,这是不可能的,因为DST和其他问题没有从本地时间到UTC时间的唯一转换。


当前回答

简单的

我是这样做的:

>>> utc_delta = datetime.utcnow()-datetime.now()
>>> utc_time = datetime(2008, 9, 17, 14, 2, 0) + utc_delta
>>> print(utc_time)
2008-09-17 19:01:59.999996

的实现

如果你想更花哨一点,你可以把它变成一个函子:

class to_utc():
    utc_delta = datetime.utcnow() - datetime.now()

    def __call__(cls, t):
        return t + cls.utc_delta

结果:

>>> utc_converter = to_utc()
>>> print(utc_converter(datetime(2008, 9, 17, 14, 2, 0)))
2008-09-17 19:01:59.999996

其他回答

我在我的一个项目中有这样的代码:

from datetime import datetime
## datetime.timezone works in newer versions of python
try:
    from datetime import timezone
    utc_tz = timezone.utc
except:
    import pytz
    utc_tz = pytz.utc

def _to_utc_date_string(ts):
    # type (Union[date,datetime]]) -> str
    """coerce datetimes to UTC (assume localtime if nothing is given)"""
    if (isinstance(ts, datetime)):
        try:
            ## in python 3.6 and higher, ts.astimezone() will assume a
            ## naive timestamp is localtime (and so do we)
            ts = ts.astimezone(utc_tz)
        except:
            ## in python 2.7 and 3.5, ts.astimezone() will fail on
            ## naive timestamps, but we'd like to assume they are
            ## localtime
            import tzlocal
            ts = tzlocal.get_localzone().localize(ts).astimezone(utc_tz)
    return ts.strftime("%Y%m%dT%H%M%SZ")

Python 3.6以来可用的选项:datetime.astimezone(tz=None)可用于获取表示本地时间的可感知datetime对象(docs)。然后可以很容易地将其转换为UTC。

from datetime import datetime, timezone
s = "2008-09-17 14:02:00"

# to datetime object:
dt = datetime.fromisoformat(s) # Python 3.7

# I'm on time zone Europe/Berlin; CEST/UTC+2 during summer 2008
dt = dt.astimezone()
print(dt)
# 2008-09-17 14:02:00+02:00

# ...and to UTC:
dtutc = dt.astimezone(timezone.utc)
print(dtutc)
# 2008-09-17 12:02:00+00:00

注意:虽然描述的到UTC的转换工作得非常好,但.astimezone()将datetime对象的tzinfo设置为timedelta派生的时区-所以不要期望从它得到任何“dst感知”。注意这里的时间增量运算。当然,除非您先转换为UTC。 相关:获取Windows上的本地时区名称(Python 3.9 zoneinfo)

注意:从2020年开始,你不应该使用.utcnow()或.utcfromtimestamp(xxx)。当您已经迁移到python3时,您应该使用时区感知的datetime对象。

>>> from datetime import timezone
>>> 
>>> # alternative to '.utcnow()'
>>> dt_now = datetime.datetime.now(datetime.timezone.utc)
>>>
>>> # alternative to '.utcfromtimestamp()'
>>> dt_ts = datetime.fromtimestamp(1571595618.0, tz=timezone.utc)

详情见:https://blog.ganssle.io/articles/2019/11/utcnow.html

最初的答案(2010年):

datetime模块的utcnow()函数可用于获取当前UTC时间。

>>> import datetime
>>> utc_datetime = datetime.datetime.utcnow()
>>> utc_datetime.strftime("%Y-%m-%d %H:%M:%S")
'2010-02-01 06:59:19'

正如Tom上面提到的链接:http://lucumr.pocoo.org/2011/7/15/eppur-si-muove/所说:

UTC是一个没有夏令时的时区,但仍然是一个时区 过去没有配置更改。 始终以UTC度量和存储时间。 如果你需要记录时间是在哪里花费的,分开存储。 不要保存本地时间+时区信息!

注意—如果您的任何数据位于使用夏令时的地区,请使用pytz并查看John Millikin的答案。

如果你想从一个给定的字符串中获得UTC时间,并且你足够幸运地处于世界上一个不使用夏令时的地区,或者你的数据只与UTC时间有偏移,而没有应用夏令时:

——>使用当地时间作为偏移值的基础:

>>> # Obtain the UTC Offset for the current system:
>>> UTC_OFFSET_TIMEDELTA = datetime.datetime.utcnow() - datetime.datetime.now()
>>> local_datetime = datetime.datetime.strptime("2008-09-17 14:04:00", "%Y-%m-%d %H:%M:%S")
>>> result_utc_datetime = local_datetime + UTC_OFFSET_TIMEDELTA
>>> result_utc_datetime.strftime("%Y-%m-%d %H:%M:%S")
'2008-09-17 04:04:00'

或者,从已知偏移量开始,使用datetime.timedelta():

>>> UTC_OFFSET = 10
>>> result_utc_datetime = local_datetime - datetime.timedelta(hours=UTC_OFFSET)
>>> result_utc_datetime.strftime("%Y-%m-%d %H:%M:%S")
'2008-09-17 04:04:00'

更新:

从python 3.2开始datetime。时区可用。你可以用下面的命令生成一个时区感知的datetime对象:

import datetime

timezone_aware_dt = datetime.datetime.now(datetime.timezone.utc)

如果你已经准备好进行时区转换,请阅读这篇文章:

https://medium.com/@eleroy/10-things-you-need-to-know-about-date-and-time-in-python-with-datetime-pytz-dateutil-timedelta-309bfbafb3f7

给那些被点赞最多的答案搞糊涂的人。你可以在python中通过生成一个datetime对象将一个datetime字符串转换为utc时间,然后你可以使用astimezone(pytz.utc)来获得utc格式的datetime。

如。

假设我们有一个isoformat的本地日期时间字符串2021-09-02T19:02:00Z

现在将该字符串转换为utc datetime。首先需要使用字符串by生成datetime对象

dt = datetime.strptime(dt,'%Y-%m-%dT%H:%M:%SZ')

这将给你一个python datetime对象,然后你可以使用astimezone(pytz.utc)来获得utc datetime like

dt = datetime.strptime(dt,'%Y-%m-%dT%H:%M:%SZ') dt = dt.astimezone(pytz.utc)

这将为您提供utc格式的datetime对象,然后您可以使用dt将其转换为字符串。Y strftime(“% - % - % d % H: % m: % S”)

完整代码eg:

from datetime import datetime
import pytz

def converLocalToUTC(datetime, getString=True, format="%Y-%m-%d %H:%M:%S"):
    dt = datetime.strptime(dt,'%Y-%m-%dT%H:%M:%SZ')
    dt = dt.astimezone(pytz.utc)
    
    if getString:
        return dt.strftime(format)
    return dt

然后你可以称它为

converLocalToUTC(“2021 - 09 - 02 t19:02:00z”)

求助于 https://stackoverflow.com/a/79877/7756843

如果你已经有了一个datetime对象my_dt,你可以用以下方法将其更改为UTC:

datetime.datetime.utcfromtimestamp(my_dt.timestamp())