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

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

3 ответов


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

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

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


нет, нет никакого способа, чтобы позвонить 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: to 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'))

взгляните на пакета 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 будет отображаться в указанном часовом поясе

для имени месяца в языковом стандарте use календарь например :

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

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