80

Python でRFC 3339 UTC タイムスタンプを生成しようとしています。これまでのところ、次のことができました。

>>> d = datetime.datetime.now()
>>> print d.isoformat('T')
2011-12-18T20:46:00.392227

私の問題は、UTC オフセットの設定にあります。

docsによると、 classmethoddatetime.now([tz])は、オプションのtz引数 where tz must be an instance of a class tzinfo subclass、およびdatetime.tzinfoisを取りますan abstract base class for time zone information objects.

ここで迷ってしまいます。なぜ tzinfo は抽象クラスなのですか? また、どのように実装すればよいのでしょうか?


(注: PHP では と同じくらい単純timestamp = date(DATE_RFC3339);です。そのため、Python のアプローチが非常に複雑な理由を理解できません...)

4

7 に答える 7

75

UPDATE 2021

Python 3.2ではtimezone、datetimeモジュールに追加され、タイムゾーンをUTCに簡単に割り当てることができます。

>>> import datetime
>>> n = datetime.datetime.now(datetime.timezone.utc)
>>> n.isoformat()
'2021-07-13T15:28:51.818095+00:00'

前の答え:

タイムゾーンは苦痛であり、それがおそらく彼らが日時ライブラリにタイムゾーンを含めないことを選択した理由です。

pytzを試してみてください。探しているtzinfoがあります:http: //pytz.sourceforge.net/

最初にオブジェクトを作成しdatetime、次に次のようにタイムゾーンを適用する必要があります。そうすると、.isoformat()出力に必要に応じてUTCオフセットが含まれます。

d = datetime.datetime.utcnow()
d_with_timezone = d.replace(tzinfo=pytz.UTC)
d_with_timezone.isoformat()

'2017-04-13T14:34:23.111142 + 00:00'

または、UTCを使用し、最後に「Z」(Zuluタイムゾーンの場合)をスローして、「タイムゾーン」をUTCとしてマークします。

d = datetime.datetime.utcnow() # <-- get time in UTC
print d.isoformat("T") + "Z"

'2017-04-13T14:34:23.111142Z'

于 2011-12-19T02:33:33.003 に答える
10

私は RFC3339 の日時形式にかなり苦労しましたが、date_string <=> datetime_object を両方向に変換する適切な解決策を見つけました。

2 つの異なる外部モジュールが必要です。そのうちの 1 つは一方向の変換しかできないためです (残念ながら):

最初のインストール:

sudo pip install rfc3339
sudo pip install iso8601

次に、以下を含めます。

import datetime     # for general datetime object handling
import rfc3339      # for date object -> date string
import iso8601      # for date string -> date object

どのモジュールがどの方向のものかを覚える必要がないように、2 つの単純なヘルパー関数を書きました。

def get_date_object(date_string):
  return iso8601.parse_date(date_string)

def get_date_string(date_object):
  return rfc3339.rfc3339(date_object)

コード内で次のように簡単に使用できます。

input_string = '1989-01-01T00:18:07-05:00'
test_date = get_date_object(input_string)
# >>> datetime.datetime(1989, 1, 1, 0, 18, 7, tzinfo=<FixedOffset '-05:00' datetime.timedelta(-1, 68400)>)

test_string = get_date_string(test_date)
# >>> '1989-01-01T00:18:07-05:00'

test_string is input_string # >>> True

ヘレカ!これで、日付文字列と日付文字列を使用可能な形式で簡単に (笑) 使用できるようになりました。

于 2016-03-13T00:26:43.733 に答える
0

実際、組み込みの datetimeモジュールを使用できます。@ruakh が言及しているよう に、ページにはその方法を示す例があります。tzinfoのセクションを調べると、 多くの異なる使用例を示す長い例が表示されます。RFC 3339 UTC タイムスタンプを生成する、探しているもののコードを次に示します。

from datetime import tzinfo, timedelta, datetime
import time as _time

ZERO = timedelta(0)
STDOFFSET = timedelta(seconds=-_time.timezone)
if _time.daylight:
    DSTOFFSET = timedelta(seconds=-_time.altzone)
else:
    DSTOFFSET = STDOFFSET

DSTDIFF = DSTOFFSET - STDOFFSET


class LocalTimezone(tzinfo):

    def utcoffset(self, dt):
        if self._isdst(dt):
            return DSTOFFSET
        else:
            return STDOFFSET

    def dst(self, dt):
        if self._isdst(dt):
            return DSTDIFF
        else:
            return ZERO

    def tzname(self, dt):
        return _time.tzname[self._isdst(dt)]

    def _isdst(self, dt):
        tt = (dt.year, dt.month, dt.day,
              dt.hour, dt.minute, dt.second,
              dt.weekday(), 0, 0)
        stamp = _time.mktime(tt)
        tt = _time.localtime(stamp)
        return tt.tm_isdst > 0

Local = LocalTimezone()

d = datetime.now(Local)
print d.isoformat('T')

# which returns
# 2014-04-28T15:44:45.758506-07:00
于 2016-09-09T19:46:51.037 に答える