Как напечатать дату и время в местном часовом поясе?

Скажем, у меня есть переменная t, которая установлена ​​на это:

datetime.datetime(2009, 7, 10, 18, 44, 59, 193982, tzinfo=<UTC>)

Если я скажу str(t), я получаю:

'2009-07-10 18:44:59.193982+00:00'

Как я могу получить подобную строку, кроме печатной в локальном часовом поясе, а не в формате UTC?

Ответы

Ответ 1

Подумайте, что вы должны смотреть вокруг: datetime.astimezone()

http://docs.python.org/library/datetime.html#datetime.datetime.astimezone

Также см. модуль pytz - он довольно прост в использовании - например:

eastern = timezone('US/Eastern')

http://pytz.sourceforge.net/

Пример:

from datetime import datetime
import pytz
from tzlocal import get_localzone # $ pip install tzlocal

utc_dt = datetime(2009, 7, 10, 18, 44, 59, 193982, tzinfo=pytz.utc)
print(utc_dt.astimezone(get_localzone())) # print local time
# -> 2009-07-10 14:44:59.193982-04:00

Ответ 2

Я считаю, что лучший способ сделать это - использовать класс LocalTimezone, определенный в документации datetime.tzinfo (goto http://docs.python.org/library/datetime.html#tzinfo-objects и прокрутите вниз до раздела "Пример tzinfo classes" ):

Предполагая, что Local является экземпляром LocalTimezone

t = datetime.datetime(2009, 7, 10, 18, 44, 59, 193982, tzinfo=utc)
local_t = t.astimezone(Local)

то str(local_t) дает:

'2009-07-11 04:44:59.193982+10:00'

что вы хотите.

(Примечание: это может показаться вам странным, потому что я в Новом Южном Уэльсе, Австралия, что на 10 или 11 часов раньше UTC)

Ответ 3

Этот скрипт демонстрирует несколько способов показать локальный часовой пояс с помощью astimezone():

#!/usr/bin/env python3

import pytz
from datetime import datetime, timezone
from tzlocal import get_localzone

utc_dt = datetime.now(timezone.utc)

PST = pytz.timezone('US/Pacific')
EST = pytz.timezone('US/Eastern')
JST = pytz.timezone('Asia/Tokyo')
NZST = pytz.timezone('Pacific/Auckland')

print("Pacific time {}".format(utc_dt.astimezone(PST).isoformat()))
print("Eastern time {}".format(utc_dt.astimezone(EST).isoformat()))
print("UTC time     {}".format(utc_dt.isoformat()))
print("Japan time   {}".format(utc_dt.astimezone(JST).isoformat()))

# Use astimezone() without an argument
print("Local time   {}".format(utc_dt.astimezone().isoformat()))

# Use tzlocal get_localzone
print("Local time   {}".format(utc_dt.astimezone(get_localzone()).isoformat()))

# Explicitly create a pytz timezone object
print("Local time   {}".format(utc_dt.astimezone(NZST).isoformat()))

Выводит следующее:

$ ./timezones.py 
Pacific time 2019-02-22T17:54:14.957299-08:00
Eastern time 2019-02-22T20:54:14.957299-05:00
UTC time     2019-02-23T01:54:14.957299+00:00
Japan time   2019-02-23T10:54:14.957299+09:00
Local time   2019-02-23T14:54:14.957299+13:00
Local time   2019-02-23T14:54:14.957299+13:00
Local time   2019-02-23T14:54:14.957299+13:00

Обратите внимание, что для вызова astimezone() без объекта часового пояса по умолчанию используется локальная зона. Это означает, что вам не нужно импортировать tzlocal. Я не совсем уверен, что это всегда безопасно по сравнению с использованием tzlocal get_localzone, возможно, кто-то может прокомментировать?

Ответ 4

Как и в python 3.2, используя только стандартные библиотечные функции:

u_tm = datetime.datetime.utcfromtimestamp(0)
l_tm = datetime.datetime.fromtimestamp(0)
l_tz = datetime.timezone(l_tm - u_tm)

t = datetime.datetime(2009, 7, 10, 18, 44, 59, 193982, tzinfo=l_tz)
str(t)
'2009-07-10 18:44:59.193982-07:00'

Просто нужно использовать l_tm - u_tm или u_tm - l_tm в зависимости от того, хотите ли вы показывать как + или - часы из UTC. Я нахожусь в MST, откуда приходит -07. Разумный код должен быть в состоянии выяснить, какой способ вычесть.

И нужно только вычислить локальный часовой пояс один раз. Это не изменится. По крайней мере, пока вы не переключитесь с/на летнее время.

Ответ 5

Я написал что-то подобное на днях:

import time, datetime
def nowString():
    # we want something like '2007-10-18 14:00+0100'
    mytz="%+4.4d" % (time.timezone / -(60*60) * 100) # time.timezone counts westwards!
    dt  = datetime.datetime.now()
    dts = dt.strftime('%Y-%m-%d %H:%M')  # %Z (timezone) would be empty
    nowstring="%s%s" % (dts,mytz)
    return nowstring

Итак, интересная часть для вас - это, вероятно, линия, начинающаяся с "mytz =...". time.timezone возвращает локальный часовой пояс, хотя с противоположным знаком по сравнению с UTC. Поэтому он говорит "-3600", чтобы выразить UTC + 1.

Несмотря на свое незнание в отношении летнего времени (DST, см. комментарий), я оставляю это для людей, играющих с time.timezone.

Ответ 6

Я использую эту функцию datetime_to_local_timezone(), которая кажется чересчур запутанной, но я не нашел более простой версии функции, которая преобразует экземпляр datetime в локальный часовой пояс, как это было настроено в операционной системе, с смещением UTC, которое было фактически в это время:

import time, datetime

def datetime_to_local_timezone(dt):
    epoch = dt.timestamp() # Get POSIX timestamp of the specified datetime.
    st_time = time.localtime(epoch) #  Get struct_time for the timestamp. This will be created using the system locale and it time zone information.
    tz = datetime.timezone(datetime.timedelta(seconds = st_time.tm_gmtoff)) # Create a timezone object with the computed offset in the struct_time.

    return dt.astimezone(tz) # Move the datetime instance to the new time zone.

utc = datetime.timezone(datetime.timedelta())
dt1 = datetime.datetime(2009, 7, 10, 18, 44, 59, 193982, utc) # DST was in effect
dt2 = datetime.datetime(2009, 1, 10, 18, 44, 59, 193982, utc) # DST was not in effect

print(dt1)
print(datetime_to_local_timezone(dt1))

print(dt2)
print(datetime_to_local_timezone(dt2))

В этом примере печатаются четыре даты. В течение двух моментов времени, один в январе и один в июле 2009 года, каждый, он печатает временную метку один раз в UTC и один раз в местном часовом поясе. Здесь, где CET (UTC + 01: 00) используется зимой и CEST (UTC + 02: 00) используется летом, он печатает следующее:

2009-07-10 18:44:59.193982+00:00
2009-07-10 20:44:59.193982+02:00

2009-01-10 18:44:59.193982+00:00
2009-01-10 19:44:59.193982+01:00