web-dev-qa-db-ja.com

「datetime.datetime」オブジェクトの属性「tzinfo」は書き込み可能ではありません

データストアから取得したばかりの日時インスタンスのタイムゾーンを設定するにはどうすればよいですか?

最初に出たときはUTCです。 ESTに変更したい。

私は試みています、例えば:

class Book( db.Model ):
    creationTime = db.DateTimeProperty()

Bookを取得したら、すぐにtzinfoを設定したいと思います。

book.creationTime.tzinfo = EST

どこ 私はこの例を使用しています 私のESTオブジェクト

しかし私は得る:

 'datetime.datetime'オブジェクトの属性 'tzinfo'に書き込みできません

私はpytzとpython-dateutilを推奨する多くの回答を見てきましたが、私は本当にこの質問に対する回答を求めています。

30
bobobobo

datetimeのオブジェクトは不変であるため、それらの属性を変更することはありません-newオブジェクトをいくつかの属性を同じにして、別のものにして、必要なものに割り当てますそれを割り当てます。

つまり、あなたのケースでは、

book.creationTime.tzinfo = EST

あなたはコーディングする必要があります

book.creationTime = book.creationTime.replace(tzinfo=EST)
49
Alex Martelli

inESTの日付時刻を受け取っているが、tzinfoフィールドが設定されていない場合は、dt.replace(tzinfo=tz)を使用して時間を変更せずにtzinfo。 (データベースがこれを行っているはずです。)

UDTの日付時刻を受け取り、ESTでそれを取得したい場合は、astimezoneが必要です。 http://docs.python.org/library/datetime.html#datetime.datetime.astimezone

ほとんどの場合、データベースはデータをUDTで格納および返す必要があり、(おそらくUDT tzinfoを割り当てる場合を除いて)置換を使用する必要はありません。

6
Glenn Maynard

docs に必要なものがあります。

from datetime import tzinfo, timedelta, datetime

ZERO = timedelta(0)
HOUR = timedelta(hours=1)
DSTSTART = datetime(1, 4, 1, 2)
DSTEND = datetime(1, 10, 25, 1)

def first_sunday_on_or_after(dt):
    days_to_go = 6 - dt.weekday()
    if days_to_go:
        dt += timedelta(days_to_go)
    return dt

class USTimeZone(tzinfo):

    def __init__(self, hours, reprname, stdname, dstname):
        self.stdoffset = timedelta(hours=hours)
        self.reprname = reprname
        self.stdname = stdname
        self.dstname = dstname

    def __repr__(self):
        return self.reprname

    def tzname(self, dt):
        if self.dst(dt):
            return self.dstname
        else:
            return self.stdname

    def utcoffset(self, dt):
        return self.stdoffset + self.dst(dt)

    def dst(self, dt):
        if dt is None or dt.tzinfo is None:
            # An exception may be sensible here, in one or both cases.
            # It depends on how you want to treat them.  The default
            # fromutc() implementation (called by the default astimezone()
            # implementation) passes a datetime with dt.tzinfo is self.
            return ZERO
        assert dt.tzinfo is self

        # Find first Sunday in April & the last in October.
        start = first_sunday_on_or_after(DSTSTART.replace(year=dt.year))
        end = first_sunday_on_or_after(DSTEND.replace(year=dt.year))

        # Can't compare naive to aware objects, so strip the timezone from
        # dt first.
        if start <= dt.replace(tzinfo=None) < end:
            return HOUR
        else:
            return ZERO

now = datetime.now()
print now
print now.tzinfo

Eastern = USTimeZone(-5, 'Eastern', 'EST', 'EDT')
now_tz_aware = now.replace(tzinfo=Eastern)
print now_tz_aware

出力:

2010-01-18 17:08:02.741482
None
2010-01-18 17:08:02.741482-05:00
0
telliott99