如何在Python中将本地时间转换为UTC?

Tom*_*Tom 284 python datetime utc localtime

如何将本地时间的日期时间字符串转换为UTC时间字符串

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

澄清:例如,如果我2008-09-17 14:02:00在我的本地时区(+10)中,我想生成一个具有相同UTC时间的字符串:2008-09-17 04:02:00.

此外,请访问http://lucumr.pocoo.org/2011/7/15/eppur-si-muove/,请注意,一般情况下这不可能与DST和其他问题一样,因此从当地时间到UTC时间.

Joh*_*kin 252

首先,将字符串解析为一个天真的日期时间对象.这是datetime.datetime没有附加时区信息的实例.有关datetime.strptime解析日期字符串的信息,请参阅文档.

使用该pytz模块,该模块附带完整的时区列表+ UTC.弄清楚当地时区是什么,从中构建时区对象,并操纵它并将其附加到天真的日期时间.

最后,使用datetime.astimezone()方法将datetime转换为UTC.

源代码,使用当地时区"America/Los_Angeles",字符串"2001-2-3 10:11:12":

import pytz, datetime
local = pytz.timezone ("America/Los_Angeles")
naive = datetime.datetime.strptime ("2001-2-3 10:11:12", "%Y-%m-%d %H:%M:%S")
local_dt = local.localize(naive, is_dst=None)
utc_dt = local_dt.astimezone(pytz.utc)
Run Code Online (Sandbox Code Playgroud)

从那里,您可以strftime()根据需要使用该方法格式化UTC日期时间:

utc_dt.strftime ("%Y-%m-%d %H:%M:%S")
Run Code Online (Sandbox Code Playgroud)

  • 请使用以下代码编辑您的答案:[code] local.localize(naive)[/ code]参见文档:[link](http://pytz.sourceforge.net/#localized-times-and-date-arithmetic )不幸的是,对于许多时区,使用标准日期时间构造函数的'tzinfo参数''与pytz不兼容.>>> datetime(2002,10,27,12,0,0,tzinfo = amsterdam).strftime(fmt)'2002-10-27 12:00:00 AMT + 0020' (7认同)
  • 显然,“弄清楚本地时区是什么”这一步骤证明了[听起来很难](http://regebro.wordpress.com/2008/05/13/thanks-for-the-testing-help-conclusions/)(几乎是不可能的)。 (2认同)
  • 按照@SamStoelinga 的建议使用 local.localize,否则您将不会考虑夏令时。 (2认同)
  • 您可以使用[使用`tzlocal.get_localzone()`将本地时区设为`pytz` tzinfo对象](http://stackoverflow.com/q/13218506/4279) (2认同)
  • 请注意,从 python 3.9 (`zoneinfo`) 或 3.6+ (`backports.zoneinfo`) 开始,可以将 pytz 替换为 IANA 时区 (2认同)

mon*_*kut 134

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'
Run Code Online (Sandbox Code Playgroud)

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

UTC是一个没有夏令时的时区,过去仍然没有配置更改的时区.

始终以UTC格式测量和存储时间.

如果您需要记录拍摄时间,请单独存储. 不要存储当地时间+时区信息!

注意 - 如果您的任何数据位于使用DST的区域,请使用pytz并查看John Millikin的答案.

如果您想从给定的字符串中获取UTC时间,并且您有幸在世界上不使用DST的区域中,或者您的数据仅在未应用DST的情况下偏离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'
Run Code Online (Sandbox Code Playgroud)

- >或者,从已知的偏移量,使用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'
Run Code Online (Sandbox Code Playgroud)

如果您准备好接受时区转换,请阅读以下内容:

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

  • 这只会转换当前时间,我需要占用任何给定时间(作为字符串)并转换为UTC. (7认同)
  • 它仅适用于当前时间,因为过去和未来的时间戳可能由于DST而具有不同的UTC偏移. (4认同)
  • 对 utcnow() 和 now() 的调用之间存在零星的增量。这是一个危险的代码行,以后可能会导致奇怪的错误,例如“tzinfo.utcoffset() 必须返回整数分钟”等等。 (2认同)

Tom*_*Tom 62

谢谢@rofly,从字符串到字符串的完整转换如下:

time.strftime("%Y-%m-%d %H:%M:%S", 
              time.gmtime(time.mktime(time.strptime("2008-09-17 14:04:00", 
                                                    "%Y-%m-%d %H:%M:%S"))))
Run Code Online (Sandbox Code Playgroud)

我的time/ calendar函数摘要:

time.strptime
string - > tuple(未应用时区,因此匹配字符串)

time.mktime
本地时间元组 - >自纪元以来的秒数(始终是当地时间)

time.gmtime
自纪元以来的秒数 - > UTC中的元组

calendar.timegm
UTC中的元组 - >自纪元以来的秒数

time.localtime
自纪元以来的秒数 - >当地时区的元组

  • 在DST过渡期间,它有50%的可能性失败.请参阅[本地时间问题](http://pytz.sourceforge.net/#problems-with-localtime) (3认同)
  • `(总是本地时间)`似乎是错误的:mktime()的输入是本地时间,输出是自纪元以来的秒数(`1970-01-01 00:00:00 +0000(UTC)`)取决于时区. (2认同)

aka*_*ola 36

以下是常见Python时间转换的摘要.

有些方法会丢弃几秒钟,并标有(s).ts = (d - epoch) / unit可以使用显式公式(感谢jfs).

  • struct_time(UTC)→POSIX (S) :
    calendar.timegm(struct_time)
  • 朴素的日期时间(本地)→POSIX (S) :(
    calendar.timegm(stz.localize(dt, is_dst=None).utctimetuple())
    期间DST转换的异常,请参阅JFS评论)
  • 朴素的日期时间(UTC)→POSIX (S) :
    calendar.timegm(dt.utctimetuple())
  • 知道日期时间→POSIX (S) :
    calendar.timegm(dt.utctimetuple())
  • POSIX→struct_time(UTC,s):(
    time.gmtime(t)
    参见jfs的评论)
  • 初始日期时间(本地)→struct_time(UTC,s):(
    stz.localize(dt, is_dst=None).utctimetuple()
    DST转换期间的异常,请参阅jfs的注释)
  • 天使日期时间(UTC)→struct_time(UTC,s):
    dt.utctimetuple()
  • 意识到datetime→struct_time(UTC,s):
    dt.utctimetuple()
  • POSIX→Naïvedatetime(local):(
    datetime.fromtimestamp(t, None)
    在某些条件下可能会失败,请参阅下面的jfs注释)
  • struct_time(UTC)→Naïvedatetime(local,s):(
    datetime.datetime(struct_time[:6], tzinfo=UTC).astimezone(tz).replace(tzinfo=None)
    不能代表闰秒,请参阅jfs的注释)
  • 天用日期时间(UTC)→天用日期时间(本地):
    dt.replace(tzinfo=UTC).astimezone(tz).replace(tzinfo=None)
  • 意识到日期时间→天使日期时间(本地):
    dt.astimezone(tz).replace(tzinfo=None)
  • POSIX→Naïvedatetime(UTC):
    datetime.utcfromtimestamp(t)
  • struct_time(UTC)→Naïvedatetime(UTC,s):(
    datetime.datetime(*struct_time[:6])
    不能代表闰秒,请参阅jfs的注释)
  • 初始日期时间(本地)→天使日期时间(UTC):(
    stz.localize(dt, is_dst=None).astimezone(UTC).replace(tzinfo=None)
    DST转换期间的异常,请参阅jfs的注释)
  • 意识到日期时间→天使日期时间(UTC):
    dt.astimezone(UTC).replace(tzinfo=None)
  • POSIX→感知日期时间:(
    datetime.fromtimestamp(t, tz)
    对于非pytz时区可能会失败)
  • struct_time(UTC)→感知日期时间(S) :(
    datetime.datetime(struct_time[:6], tzinfo=UTC).astimezone(tz)
    不能代表闰秒,看到JFS评论)
  • 天会日期时间(本地)→意识到日期时间:(
    stz.localize(dt, is_dst=None)
    DST转换期间的异常,请参阅jfs的评论)
  • 天真日期时间(UTC)→意识到日期时间:
    dt.replace(tzinfo=UTC)

资料来源:taaviburns.ca


Chu*_*ebs 25

def local_to_utc(t):
    secs = time.mktime(t)
    return time.gmtime(secs)

def utc_to_local(t):
    secs = calendar.timegm(t)
    return time.localtime(secs)
Run Code Online (Sandbox Code Playgroud)

资料来源:http://feihonghsu.blogspot.com/2008/02/converting-from-local-time-to-utc.html

bd808的示例用法:如果您的源是datetime.datetime对象t,请调用:

local_to_utc(t.timetuple())
Run Code Online (Sandbox Code Playgroud)

  • 如果你的源是datetime.datetime对象`t`,请调用:local_to_utc(t.timetuple()) (5认同)
  • `.timetuple()`调用将`tm_isdst`设置为`-1`; 在DST过渡期间,有50%的可能性`mktime()`失败. (2认同)

Yar*_*rin 19

我对dateutil好运(在其他相关问题上广泛推荐使用SO):

from datetime import *
from dateutil import *
from dateutil.tz import *

# METHOD 1: Hardcode zones:
utc_zone = tz.gettz('UTC')
local_zone = tz.gettz('America/Chicago')
# METHOD 2: Auto-detect zones:
utc_zone = tz.tzutc()
local_zone = tz.tzlocal()

# Convert time string to datetime
local_time = datetime.strptime("2008-09-17 14:02:00", '%Y-%m-%d %H:%M:%S')

# Tell the datetime object that it's in local time zone since 
# datetime objects are 'naive' by default
local_time = local_time.replace(tzinfo=local_zone)
# Convert time to UTC
utc_time = local_time.astimezone(utc_zone)
# Generate UTC time string
utc_string = utc_time.strftime('%Y-%m-%d %H:%M:%S')
Run Code Online (Sandbox Code Playgroud)

(代码源自将UTC UTC日期时间字符串转换为本地日期时间的答案)


Pau*_*ius 18

pytz的另一个例子,但包括localize(),这节省了我的一天.

import pytz, datetime
utc = pytz.utc
fmt = '%Y-%m-%d %H:%M:%S'
amsterdam = pytz.timezone('Europe/Amsterdam')

dt = datetime.datetime.strptime("2012-04-06 10:00:00", fmt)
am_dt = amsterdam.localize(dt)
print am_dt.astimezone(utc).strftime(fmt)
'2012-04-06 08:00:00'
Run Code Online (Sandbox Code Playgroud)


小智 12

我用python-dateutil取得了最大的成功:

from dateutil import tz

def datetime_to_utc(date):
    """Returns date in UTC w/o tzinfo"""
    return date.astimezone(tz.gettz('UTC')).replace(tzinfo=None) if date.tzinfo else date
Run Code Online (Sandbox Code Playgroud)

  • [`dateutil`可能在过去的日期中失败](http://stackoverflow.com/questions/79797/how-do-i-convert-local-time-to-utc-in-python/2175170#comment30454158_8563126) (2认同)

小智 7

import time

import datetime

def Local2UTC(LocalTime):

    EpochSecond = time.mktime(LocalTime.timetuple())
    utcTime = datetime.datetime.utcfromtimestamp(EpochSecond)

    return utcTime

>>> LocalTime = datetime.datetime.now()

>>> UTCTime = Local2UTC(LocalTime)

>>> LocalTime.ctime()

'Thu Feb  3 22:33:46 2011'

>>> UTCTime.ctime()

'Fri Feb  4 05:33:46 2011'
Run Code Online (Sandbox Code Playgroud)


Bry*_*yce 7

这是Python3.9中本机zoneinfo模块的示例:

from datetime import datetime
from zoneinfo import ZoneInfo

# Get timezone we're trying to convert from
local_tz = ZoneInfo("America/New_York")
# UTC timezone
utc_tz = ZoneInfo("UTC")

dt = datetime.strptime("2021-09-20 17:20:00","%Y-%m-%d %H:%M:%S")
dt = dt.replace(tzinfo=local_tz)
dt_utc = dt.astimezone(utc_tz)

print(dt.strftime("%Y-%m-%d %H:%M:%S"))
print(dt_utc.strftime("%Y-%m-%d %H:%M:%S"))
Run Code Online (Sandbox Code Playgroud)

dt.astimezone()在您要转换的时区不反映系统本地时区的情况下,这可能比仅使用更可取。不必依赖外部库也很好。

注意:这可能不适用于 Windows 系统,因为zoneinfo 依赖于可能不存在的IANA 数据库。可以安装 tzdata 包作为解决方法它是第一方包,但不在标准库中。


小智 6

简单的

我是这样做的:

>>> 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
Run Code Online (Sandbox Code Playgroud)

花式实施

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

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

    def __call__(cls, t):
        return t + cls.utc_delta
Run Code Online (Sandbox Code Playgroud)

结果:

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

  • 这不适用于夏令时 - 例如,如果当前是夏季并且您要转换的日期是冬季。问题是关于转换存储为字符串的日期...... (6认同)
  • 供参考。这种方法的最大问题(除了夏令时)是增量将关闭的几毫秒。我所有的日历邀请都显示为关闭 1 分钟。 (2认同)

MrF*_*pes 6

此线程似乎缺少自 Python 3.6 起可用的选项:datetime.astimezone(tz=None)can be used to get an aware datetime object representation of local time (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
Run Code Online (Sandbox Code Playgroud)

但是有一个警告,请参阅astimezone(None) 提供了知道日期时间,不知道 DST

  • 由于所有其他答案都相当旧,并且这是没有任何外部包的解决方案(日期时间是内置的),因此这个答案值得更多的信任。 (3认同)

use*_*042 5

如果您更喜欢datetime.datetime:

dt = datetime.strptime("2008-09-17 14:04:00","%Y-%m-%d %H:%M:%S")
utc_struct_time = time.gmtime(time.mktime(dt.timetuple()))
utc_dt = datetime.fromtimestamp(time.mktime(utc_struct_time))
print dt.strftime("%Y-%m-%d %H:%M:%S")
Run Code Online (Sandbox Code Playgroud)

  • 这是不正确的。如果本地时区不是UTC,则失败。“ mktime()”期望本地时间作为输入。fromtimestamp()返回本地时间,而不是utc。如果您对其进行修复,则会看到这些[其他问题(例如,“ dateutil”,仅stdlib解决方案可能会失败)](http://stackoverflow.com/questions/79797/how-do-i-convert-local-time-to -utc-in-python#comment30454158_8563126) (2认同)

归档时间:

查看次数:

377612 次

最近记录:

5 年,10 月 前