Как мне strftime объект даты в другой локали?

У меня есть объект даты в python, и мне нужно сгенерировать отметку времени в локали C для устаревшей системы, используя коды %a (день недели) и %b (месяц). Однако я не хочу менять локаль приложения, поскольку другие части должны учитывать текущую локаль пользователя. Есть ли способ вызвать strftime() с определенной локалью?


person MagerValp    schedule 03.09.2013    source источник


Ответы (3)


Пример, данный Робом, великолепен, но не является потокобезопасным. Вот версия, которая работает с потоками:

import locale
import threading

from datetime import datetime
from contextlib import contextmanager


LOCALE_LOCK = threading.Lock()

@contextmanager
def setlocale(name):
    with LOCALE_LOCK:
        saved = locale.setlocale(locale.LC_ALL)
        try:
            yield locale.setlocale(locale.LC_ALL, name)
        finally:
            locale.setlocale(locale.LC_ALL, saved)

# Let's set a non-US locale
locale.setlocale(locale.LC_ALL, 'de_DE.UTF-8')

# Example to write a formatted English date
with setlocale('C'):
    print(datetime.now().strftime('%a, %b')) # e.g. => "Thu, Jun"

# Example to read a formatted English date
with setlocale('C'):
    mydate = datetime.strptime('Thu, Jun', '%a, %b')

Он создает потокобезопасный диспетчер контекста, используя глобальную блокировку, и позволяет вам иметь несколько потоков, выполняющих код, зависящий от локали, с помощью LOCALE_LOCK. Он также обрабатывает исключения из оператора yield, чтобы всегда восстанавливалась исходная локаль.

person Daniel    schedule 05.06.2014
comment
Обратите внимание, что для успешного выполнения setlocale() необходимо, чтобы эта локаль была установлена ​​на системном уровне. На что, вообще говоря, полагаться нельзя. Для всего, что означает возможность локализации, вам нужно держаться подальше от strftime и использовать подходящую библиотеку i18n, например Babel. - person ddaa; 20.08.2014
comment
это опечатка на saved = locale.setlocale(... ? - person vonPetrushev; 20.10.2014
comment
Я не верю, что это опечатка - если вторым аргументом setlocale является None, вы получите текущую установленную локаль. - person Daniel; 21.10.2014
comment
@ddaa: Спасибо за это предложение. Действительно, нельзя полагаться на то, что они устанавливаются, даже если вы живете в стране, для которой пытаетесь использовать локаль. Вавилон - это действительно правильный путь, и это должен быть отдельный ответ. - person WhyNotHugo; 15.03.2015
comment
Будет ли он по-прежнему yield, если setlocale бросит? Разве это не испортит конструкцию with? (например, преждевременное поднятие StopIteration и т. д.) (И, кстати, классный ответ, там много всего происходит. Я думал, что проблема с локалью для каждого процесса не может быть так просто отправлена.) - person Tomasz Gandor; 25.04.2016
comment
Разве это не блокирует другие потоки от одновременной установки локали? Он ничего не делает, чтобы изолировать глобальное изменение от воздействия на другой код с учетом локали, который не использует эту конструкцию. большинство из них этого не сделали бы, поскольку обычно они просто используют настройку локали в начале программы. Или я что-то упускаю? - person altendky; 07.04.2021

Нет, нельзя вызвать strftime() с определенной локалью.

Предполагая, что ваше приложение не является многопоточным, сохраните и восстановите существующую локаль и установите для нее значение 'C' при вызове strftime.

#! /usr/bin/python3
import time
import locale


def get_c_locale_abbrev():
  lc = locale.setlocale(locale.LC_TIME)
  try:
    locale.setlocale(locale.LC_TIME, "C")
    return time.strftime("%a-%b")
  finally:
    locale.setlocale(locale.LC_TIME, lc)

# Let's suppose that we're french
locale.setlocale(locale.LC_ALL, 'fr_FR.utf8')

# Should print french, english, then french
print(time.strftime('%a-%b'))
print(get_c_locale_abbrev())
print(time.strftime('%a-%b'))

Если вы предпочитаете with: вместо try:-finally:, вы можете создать контекстный менеджер:

#! /usr/bin/python3
import time
import locale
import contextlib

@contextlib.contextmanager
def setlocale(*args, **kw):
  saved = locale.setlocale(locale.LC_ALL)
  yield locale.setlocale(*args, **kw)
  locale.setlocale(locale.LC_ALL, saved)

def get_c_locale_abbrev():
  with setlocale(locale.LC_TIME, "C"):
    return time.strftime("%a-%b")

# Let's suppose that we're french
locale.setlocale(locale.LC_ALL, 'fr_FR.utf8')

# Should print french, english, then french
print(time.strftime('%a-%b'))
print(get_c_locale_abbrev())
print(time.strftime('%a-%b'))
person Robᵩ    schedule 03.09.2013
comment
это локаль во всем сценарии, я думаю, так что будьте осторожны - person Philippe T.; 03.09.2013
comment
Да, locale.setlocale() влияет на всю программу. Блок try-finally или блок with устанавливает и восстанавливает локаль, чтобы не мешать остальной части программы. - person Robᵩ; 03.09.2013

взгляните на пакет pytz

вы можете использовать вот так

import pytz
UTC = pytz.timezone('UTC') # utc
fr = pytz.timezone('Europe/Paris') #your local
from datetime import datetime
date = datetime.now(fr)
dateUTC = date.astimezone(UTC)

strftime будет отображаться в указанном часовом поясе

для названия месяца в локали используйте calendar, например:

import calendar
print calendar.month_name[dateUTC.month] #will print in the locale

более глубоко изучить календарь для получения дополнительной информации

person Philippe T.    schedule 03.09.2013
comment
Это интересно, но мне нужно создать отметку даты с английскими словами для дней недели (%a) и месяцев (%b), а не менять часовой пояс. Может ли pytz сделать это тоже? - person MagerValp; 03.09.2013
comment
взгляните на календарь docs.python.org/2/library/calendar .html#module-calendar и, возможно, метод calendar.month_name - person Philippe T.; 03.09.2013
comment
Опять же, это не то, чего хочет ОП. calendar.month_name Массив, представляющий месяцы года в текущей локали. - person Dima Tisnek; 11.03.2015