如何将本地时间的日期时间字符串转换为 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 时间的唯一转换。
首先,将字符串解析为一个简单的日期时间对象。这是没有附加时区信息的 datetime.datetime
实例。查看它的 documentation。
使用 pytz
模块,它带有完整的时区列表 + UTC。弄清楚本地时区是什么,从中构造一个时区对象,然后对其进行操作并将其附加到天真的日期时间。
最后,使用 datetime.astimezone()
方法将日期时间转换为 UTC。
源代码,使用本地时区“America/Los_Angeles”,字符串“2001-2-3 10:11:12”:
from datetime import datetime
import pytz
local = pytz.timezone("America/Los_Angeles")
naive = 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)
从那里,您可以根据需要使用 strftime()
方法来格式化 UTC 日期时间:
utc_dt.strftime("%Y-%m-%d %H:%M:%S")
注意 -- 自 2020 年起,您不应使用 .utcnow()
或 .utcfromtimestamp(xxx)
。由于您可能已经转向 python3,您应该使用时区感知日期时间对象。
>>> 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'
正如汤姆上面提到的链接:http://lucumr.pocoo.org/2011/7/15/eppur-si-muove/ 所说:
UTC 是一个没有夏令时的时区,并且仍然是一个过去没有配置更改的时区。始终以 UTC 测量和存储时间。如果您需要记录所用时间的位置,请单独存储。不要存储本地时间+时区信息!
注意 - 如果您的任何数据位于使用 DST 的区域,请使用 pytz
并查看 John Millikin 的答案。
如果您想从给定的字符串中获取 UTC 时间,并且您很幸运地处于世界上不使用 DST 的地区,或者您的数据仅与 UTC 偏移而未应用 DST:
--> 使用本地时间作为偏移值的基础:
>>> # 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.timezone
可用。您可以使用以下命令生成时区感知日期时间对象:
import datetime
timezone_aware_dt = datetime.datetime.now(datetime.timezone.utc)
如果您准备好进行时区转换,请阅读以下内容:
tzinfo.utcoffset() must return a whole number of minutes
等等。
感谢@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"))))
我对 time
/calendar
函数的总结:
time.strptime
字符串 -->元组(未应用时区,因此匹配字符串)
time.mktime
本地时间元组 -->自纪元以来的秒数(始终为当地时间)
time.gmtime
自纪元以来的秒数 --> UTC 元组
和
calendar.timegm
UTC 元组 -->自纪元以来的秒数
time.localtime
自纪元以来的秒数 -->本地时区的元组
(always local time)
似乎是错误的: mktime() 的输入是本地时间,输出是自纪元 (1970-01-01 00:00:00 +0000 (UTC)
) 以来的秒数,不依赖于时区。
这里总结了常见的 Python 时间转换。
有些方法会减少几分之一秒,并用 (s) 标记。可以改用 ts = (d - epoch) / unit
等显式公式(感谢 jfs)。
struct_time (UTC) → POSIX (s): calendar.timegm(struct_time)
Naïve datetime (local) → 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 的评论)
Naïve datetime (local) → 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ïve datetime (local): datetime.fromtimestamp(t, None) (在某些情况下可能会失败,请参阅下面 jfs 的评论)
struct_time (UTC) → Naïve datetime (local, s): datetime.datetime(struct_time[:6], tzinfo=UTC).astimezone(tz).replace(tzinfo=None) (不能代表闰秒,见评论来自杰夫斯)
朴素日期时间(UTC)→ 朴素日期时间(本地):dt.replace(tzinfo=UTC).astimezone(tz).replace(tzinfo=None)
Aware datetime → Naïve datetime (local): dt.astimezone(tz).replace(tzinfo=None)
POSIX → 朴素日期时间 (UTC):datetime.utcfromtimestamp(t)
struct_time (UTC) → Naïve datetime (UTC, s): datetime.datetime(*struct_time[:6]) (不能代表闰秒,见 jfs 的评论)
Naïve datetime (local) → Naïve datetime (UTC): stz.localize(dt, is_dst=None).astimezone(UTC).replace(tzinfo=None) (DST 转换期间的异常,请参阅 jfs 的评论)
知道日期时间 → 天真的日期时间 (UTC):dt.astimezone(UTC).replace(tzinfo=None)
POSIX → Aware datetime: datetime.fromtimestamp(t, tz) (对于非 pytz 时区可能会失败)
struct_time (UTC) → Aware datetime (s): datetime.datetime(struct_time[:6], tzinfo=UTC).astimezone(tz) (不能代表闰秒,见 jfs 的评论)
Naïve datetime (local) → Aware datetime: stz.localize(dt, is_dst=None) (DST 转换期间的异常,请参阅 jfs 的评论)
天真的日期时间 (UTC) → 知道日期时间:dt.replace(tzinfo=UTC)
t
和 C 库无法访问给定平台上的 tz 数据库,则 fromtimestamp(t, None)
可能会失败。您可以使用 tzlocal.get_localzone()
以可移植的方式提供 tzdata。 (2) fromtimestamp(t, tz)
对于非 pytz 时区可能会失败。 (3) datetime(*struct_time[:6])
你缺少 *
。 (4) 基于 timegm()
, utctimetuple()
, struct_time
的解决方案会下降几分之一秒。您可以改用明确的公式,例如:ts = (d - epoch) / unit
。
stz.localize(dt, is_dst=None)
为不明确的或不存在的本地时间引发异常,例如,在 DST 转换期间。为避免该异常,请使用可能返回不精确结果的 stz.normalize(stz.localize(dt))
。 (6) datetime() 不能代表闰秒。 Convert struct_time
to "seconds since epoch" as a workaround first. (7) time.gmtime(t)
与 calendar.timegm()
不同,如果使用“正确”时区,则可能需要非 POSIX 输入。如果输入是 POSIX,则使用显式公式:gmtime = lambda t: datetime(1970,1,1, tzinfo=utc) + timedelta(seconds=t)
<table>
标签。抱歉。这是有意设计的。如果您需要一个快速而肮脏的“表格”,请使用 <pre>
和 ASCII 布局。”我不相信 ASCII 表会比上面的列表更具可读性。
我对 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')
(代码来源于此对 Convert UTC datetime string to local datetime 的回答)
dateutil
在过去的日期可能会失败,而系统的实施不使用历史时区数据库(尤其是 Windows)。可以改用 pytz
+ tzlocal
。
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)
来源:http://feihonghsu.blogspot.com/2008/02/converting-from-local-time-to-utc.html
bd808 的示例用法:如果您的源是 datetime.datetime
对象 t
,则调用如下:
local_to_utc(t.timetuple())
t
,则调用为:local_to_utc(t.timetuple())
.timetuple()
调用集 tm_isdst
到 -1
;在 DST 转换期间有 50% 的机会 mktime()
失败。
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'
自 Python 3.6 起可用的选项:datetime.astimezone(tz=None)
可用于获取表示本地时间 (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 意识”。
我在 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
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'
mktime(dt.timetuple())
can fail during DST transition。文档中有 LocalTimezone
(它适用于最新的时区定义,但对于过去的日期可能会失败(与 DST 无关))
如果您更喜欢 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")
mktime()
期望本地时间作为输入。 fromtimestamp()
返回当地时间,而不是 UTC。如果您修复它,请查看这些 additional issues (like dateutil
, stdlib-only solution may fail)
简单的
我是这样做的:
>>> 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
怎么样 -
time.strftime("%Y-%m-%dT%H:%M:%SZ", time.gmtime(seconds))
如果 seconds 为 None
,则它将本地时间转换为 UTC 时间,否则将传入的时间转换为 UTC。
你可以这样做:
>>> from time import strftime, gmtime, localtime
>>> strftime('%H:%M:%S', gmtime()) #UTC time
>>> strftime('%H:%M:%S', localtime()) # localtime
以下是 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"))
在您要转换的时区不反映系统的本地时区的情况下,这可能比仅使用 dt.astimezone()
更可取。不必依赖外部库也很好。
注意:这可能不适用于 Windows 系统,因为 zoneinfo relies on an IANA database 可能不存在。可以安装 tzdata 软件包作为解决方法。它是第一方包,但不在标准库中。
用于避开夏令时等。
以上答案都没有对我特别有帮助。下面的代码适用于 GMT。
def get_utc_from_local(date_time, local_tz=None):
assert date_time.__class__.__name__ == 'datetime'
if local_tz is None:
local_tz = pytz.timezone(settings.TIME_ZONE) # Django eg, "Europe/London"
local_time = local_tz.normalize(local_tz.localize(date_time))
return local_time.astimezone(pytz.utc)
import pytz
from datetime import datetime
summer_11_am = datetime(2011, 7, 1, 11)
get_utc_from_local(summer_11_am)
>>>datetime.datetime(2011, 7, 1, 10, 0, tzinfo=<UTC>)
winter_11_am = datetime(2011, 11, 11, 11)
get_utc_from_local(winter_11_am)
>>>datetime.datetime(2011, 11, 11, 11, 0, tzinfo=<UTC>)
使用 http://crsmithdev.com/arrow/
arrowObj = arrow.Arrow.strptime('2017-02-20 10:00:00', '%Y-%m-%d %H:%M:%S' , 'US/Eastern')
arrowObj.to('UTC') or arrowObj.to('local')
这个图书馆让生活变得轻松:)
我的一个项目中有这段代码:
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.9.0 中,将本地时间 local_time
解析为 datetime.datetime
对象后,只需使用 local_time.astimezone(datetime.timezone.utc)
。
我在另一个问题 here 上找到了最佳答案。它只使用 python 内置库,不需要您输入本地时区(在我的情况下是一个要求)
import time
import calendar
local_time = time.strptime("2018-12-13T09:32:00.000", "%Y-%m-%dT%H:%M:%S.%f")
local_seconds = time.mktime(local_time)
utc_time = time.gmtime(local_seconds)
我在这里重新发布答案,因为这个问题会根据搜索关键字在谷歌中弹出,而不是链接问题。
如果您已经有一个日期时间对象 my_dt
,您可以将其更改为 UTC:
datetime.datetime.utcfromtimestamp(my_dt.timestamp())
对于任何对最受好评的答案感到困惑的人。您可以通过生成 datetime 对象将 datetime 字符串转换为 python 中的 utc 时间,然后您可以使用 astimezone(pytz.utc) 以 utc 获取 datetime。
例如。
假设我们在 isoformat 中有本地日期时间字符串为 2021-09-02T19:02:00Z
现在将此字符串转换为 utc 日期时间。我们首先需要使用这个字符串生成日期时间对象
dt = datetime.strptime(dt,'%Y-%m-%dT%H:%M:%SZ')
这将为您提供 python 日期时间对象,然后您可以使用 astimezone(pytz.utc)
来获取 UTC 日期时间,例如
dt = datetime.strptime(dt,'%Y-%m-%dT%H:%M:%SZ') dt = dt.astimezone(pytz.utc)
这将为您提供 UTC 格式的日期时间对象,然后您可以使用 dt.strftime("%Y-%m-%d %H:%M:%S")
将其转换为字符串
完整代码例如:
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-02T19:02:00Z")
得到https://stackoverflow.com/a/79877/7756843的帮助
简要,将任何 datetime
日期转换为 UTC 时间:
from datetime import datetime
def to_utc(date):
return datetime(*date.utctimetuple()[:6])
让我们用一个例子来解释。首先,我们需要从字符串创建一个 datetime
:
>>> date = datetime.strptime("11 Feb 2011 17:33:54 -0800", "%d %b %Y %H:%M:%S %z")
然后,我们可以调用该函数:
>>> to_utc(date)
datetime.datetime(2011, 2, 12, 1, 33, 54)
逐步介绍该功能的工作原理:
>>> date.utctimetuple()
time.struct_time(tm_year=2011, tm_mon=2, tm_mday=12, tm_hour=1, tm_min=33, tm_sec=54, tm_wday=5, tm_yday=43, tm_isdst=0)
>>> date.utctimetuple()[:6]
(2011, 2, 12, 1, 33, 54)
>>> datetime(*date.utctimetuple()[:6])
datetime.datetime(2011, 2, 12, 1, 33, 54)
在python3中:
pip install python-dateutil
from dateutil.parser import tz
mydt.astimezone(tz.gettz('UTC')).replace(tzinfo=None)
from dateutil import tz
怎么样 -
time.strftime("%Y-%m-%dT%H:%M:%SZ", time.gmtime(seconds))
如果 seconds 为 None
,则它将本地时间转换为 UTC 时间,否则将传入的时间转换为 UTC。
tzlocal.get_localzone()
to get the local timezone aspytz
tzinfo object