現地時間の文字列をUTCに変換する方法は?


308

どのように私は、datetime型の変換か現地時間の文字列をする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時間。


1
mktime()メソッドは「ローカル時間」を入力として受け取るので、期待とは異なる可能性があることに注意してください。これを使用すると、すべてが台無しになりました。このリンクを
Haifeng Zhang

回答:


288

最初に、文字列を解析して単純なdatetimeオブジェクトにします。これは、datetime.datetimeタイムゾーン情報が添付されていないのインスタンスです。datetime.strptime日付文字列の解析については、ドキュメントを参照してください。

pytzタイムゾーンとUTCの完全なリストが付属しているモジュールを使用します。ローカルタイムゾーンが何であるかを理解し、そこからタイムゾーンオブジェクトを作成し、操作してナイーブな日時にアタッチします。

最後に、datetime.astimezone()メソッドを使用して日時をUTCに変換します。

文字列「2001-2-3 10:11:12」のローカルタイムゾーン「America / Los_Angeles」を使用したソースコード:

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)

そこから、strftime()メソッドを使用して、必要に応じてUTC日時をフォーマットできます。

utc_dt.strftime ("%Y-%m-%d %H:%M:%S")

7
次のコードで回答を編集してください:[code] local.localize(naive)[/ code]ドキュメントを参照してください:link 残念ながら、標準の日時コンストラクターのtzinfo引数を使用すると、多くのタイムゾーンでpytzが「機能しません」。>>> datetime(2002、10、27、12、0、0、tzinfo = amsterdam).strftime(fmt) '2002-10-27 12:00:00 AMT + 0020'
Sam Stoelinga

2
どうやら、「ローカルタイムゾーンを特定する」ステップは、実際よりも難しい(実際には不可能)ことがわかります。
Jason R. Coombs、2011

2
@SamStoelingaの提案に従ってlocal.localizeを使用します。そうしないと、夏時間を考慮しません。
EmilStenström

この回答は私にとって非常に役に立ちましたが、私が遭遇した落とし穴を指摘したいと思います。日付を指定せずに時刻を指定すると、予期しない結果が生じる可能性があります。したがって、完全なタイムスタンプを提供するようにしてください。
Steven Mercatante 2013年


145

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を使用するリージョンにある場合は、pytzJohn 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'

->または、既知のオフセットから、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)

タイムゾーン変換を行う準備ができている場合は、以下をお読みください。

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


14
これは現在の時刻のみを変換するので、任意の時間を(文字列として)取り、UTCに変換する必要があります。
トム

標準のオフセット値を使用している場合は、ここでは問題にならないと思います。local_time = utc_time + utc_offset AND utc_time = local_time-utc_offset。
monkut

5
DSTが原因で、過去と将来のタイムスタンプが異なるUTCオフセットを持つ可能性があるため、現在の時刻でのみ機能します。
アレックスB

良い点... DSTに対処する必要がある場合は、John Millikinによって言及されているように、おそらくpytzを使用する必要があります。
monkut

1
utcnow()とnow()の呼び出しの間に散発的なデルタがあります。これは、後で奇妙なエラーが発生するなどの危険なコード行ですtzinfo.utcoffset() must return a whole number of minutes
Pavel Vlasov、2012年

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"))))

time/ calendar関数の私の要約:

time.strptime
文字列->タプル(タイムゾーンは適用されないため、文字列に一致します)

time.mktime
現地時間のタプル->エポックからの秒数(常に現地時間)

time.gmtime
エポックからの秒数-> UTCのタプル

そして

calendar.timegm
UTCのタプル->エポックからの秒数

time.localtime
エポックからの秒数->ローカルタイムゾーンのタプル


4
(always local time)間違っているようです:mktime()への入力は現地時間です。出力は1970-01-01 00:00:00 +0000 (UTC)、タイムゾーンに依存しないエポック()からの秒数です。
jfs 2012

3
また、DST移行中に失敗する可能性は50%です。Localtimeの問題を
jfs

38

一般的な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()
  • 日時を認識→struct_time(UTC、s):
    dt.utctimetuple()
  • POSIX→ナイーブ日時(ローカル):(
    datetime.fromtimestamp(t, None)
    特定の条件で失敗する可能性があります。以下のjfsからのコメントを参照してください)
  • struct_time(UTC)→ナイーブ日時(ローカル、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→ナイーブ日時(UTC):
    datetime.utcfromtimestamp(t)
  • struct_time(UTC)→単純な日付時刻(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)
    非ピッツタイムゾーンでは失敗する場合があります)
  • 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


1
(1)fromtimestamp(t, None)ローカルのタイムゾーンに異なるutcオフセットがt あり、 Cライブラリが指定されたプラットフォームのtzデータベースにアクセスできない場合、失敗することがあります。tzlocal.get_localzone()ポータブルな方法でtzdataを提供するために使用できます。(2)fromtimestamp(t, tz)ピッツ以外のタイムゾーンでは失敗する可能性があります。(3)datetime(*struct_time[:6])あなたが行方不明*です。(4)、、- ベースのソリューションtimegm()utctimetuple()struct_timeほんの一瞬をドロップします。ts = (d - epoch) / unit代わりに、次のような明示的な式を使用できます。
jfs

1
(5)stz.localize(dt, is_dst=None)たとえば、DSTの移行中など、あいまいまたは存在しない現地時間の例外を発生させます。例外を回避するには、stz.normalize(stz.localize(dt))不正確な結果を返す可能性があるを使用します。(6)datetime()はうるう秒を表すことができません。最初に回避策として「エポックからの秒数」に変換struct_timeします。(7)「正しい」タイムゾーンが使用されている場合、POSIX以外の入力が期待されるtime.gmtime(t)とは異なりcalendar.timegm()ます。:入力が代わりにPOSIXの場合は、明示的な式を使用してくださいgmtime = lambda t: datetime(1970,1,1, tzinfo=utc) + timedelta(seconds=t)
JFS

これが表になっていて、読みやすくなっていることを望みます(リンクにあります)
MichaelChirico

1
@MichaelChirico、この回答は、「<table>タグは許可されません(許可されません)です。申し訳ありません。これは意図的であり、設計によるものです。迅速で汚い「テーブル」が必要な場合は<pre>、ASCIIレイアウトを使用してください。」ASCIIテーブルが上記のリストより読みやすいとは思いません。
akaihola

まあそれは残念です。とにかくあなたは私の賛成票を持っています...多分あなたの答えの中のリンクにある表を参照しますか?
MichaelChirico

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)

出典:http : //feihonghsu.blogspot.com/2008/02/converting-from-local-time-to-utc.html

bd808の使用例:ソースがdatetime.datetimeオブジェクトの場合t、次のように呼び出します。

local_to_utc(t.timetuple())

5
ソースがdatetime.datetimeオブジェクトの場合はt、次のように呼び出します。local_to_utc(t.timetuple())
bd808

2
.timetuple()をに設定tm_isdst-1ます。mktime()DSTの移行中に失敗する可能性は50%です。
jfs 2013

1
Chuckによるソリューションでは、ミリ秒の情報が失われます。
ルカ

21

私は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')

(コードは、UTCの日時文字列をローカルの日時変換するためのこの回答から導き出されました)


dateutil実装が履歴タイムゾーンデータベース(特にWindows)を使用していないシステムでは、ローカルタイムゾーンが異なるutcオフセット(DST移行とは無関係)を持っている場合、過去の日付で失敗する可能性があります。pytz+ tzlocal代わりに使用できます。
jfs 2013

@ JFSebastian-それについて聞いたことがありません-詳細情報はどこで入手できますか?
Yarin、

Windowsマシンを使用して、過去に異なるutcオフセットがあったタイムゾーンを設定します(例:Europe / Moscow)。結果をpytzと比較します。さらに、この場合の正しいAPIの使用法についてはわかりませんが、Ubuntuでも失敗するこのバグをテストできます。
jfs

出力は何ですか?出力を投稿してください。
not2qubit

18

pytzを使用したもう1つの例ですが、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'


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'

mktime(dt.timetuple())DST移行中に失敗する可能性があります。ありますLocalTimezone(これは、最新のタイムゾーンの定義のために動作しますが、DSTに関係のない過去の日付()で失敗することができます)ドキュメントに
JFS

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")

1
承知しましたが、なぜそれを好むのかわかりません。追加のインポートと私のバージョンより3つ多い関数呼び出しが必要です...
Tom

%Zと%zは空白を印刷します。
Pavel Vlasov

2
間違いです。ローカルのタイムゾーンがUTCでない場合は失敗します。mktime()入力として現地時間を想定しています。fromtimestamp()utcではなく現地時間を返します。それを修正すると、これらの追加の問題が表示されます(dateutilstdlibのみのソリューションが失敗するなど)
jfs

5

シンプルな

私はそれをこのようにしました:

>>> 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

5
これは夏時間に対しては機能しません。たとえば、現在夏で、変換する日付が冬の場合などです。そして、質問さは約...文字列として保存された日付を変換した
トム

1
ご参考までに。この方法の最大の問題(夏時間以外)は、デルタがオフになる数ミリ秒です。カレンダーの招待がすべて1分遅れて表示されます。
Nostalg.io 2018年

4

あなたはそれを行うことができます:

>>> from time import strftime, gmtime, localtime
>>> strftime('%H:%M:%S', gmtime()) #UTC time
>>> strftime('%H:%M:%S', localtime()) # localtime

3

どのように-

time.strftime("%Y-%m-%dT%H:%M:%SZ", time.gmtime(seconds))

秒の場合はNone、ローカル時間をUTC時間に変換します。それ以外の場合は、渡された時間をUTCに変換します。


2

夏時間の移動などに。

上記の答えのどれも、特に私を助けませんでした。以下のコードは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>)

2

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') 

このライブラリは人生を楽にします:)


arrow is awesome:arrow.get(datetime.now()、 'local')。to( 'utc')。naive
SteveJ

1

ここで別の質問に最良の答えを見つけました。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)

この質問は、検索キーワードに応じてリンクされた質問ではなくGoogleにポップアップ表示されるため、ここで回答を再投稿しています。


1

私のプロジェクトの1つにこのコードがあります。

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")

0

python3の場合:

pip install python-dateutil

from dateutil.parser import tz

mydt.astimezone(tz.gettz('UTC')).replace(tzinfo=None) 

ValueError:これはとValueErrorの例外が発生し、動作していないようastimezone()は、ナイーブ日時に適用することはできません
Stormsson

次のようになります from dateutil import tz
ハビエル、

-3

どのように-

time.strftime("%Y-%m-%dT%H:%M:%SZ", time.gmtime(seconds))

秒の場合はNone、ローカル時間をUTC時間に変換します。それ以外の場合は、渡された時間をUTCに変換します。


1
これは役に立たない、問題は与えられたローカルタイム文字列をutc文字列に変換することでした。
トム
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.