Comment puis-je strftime un objet date dans un endroit différent?

j'ai un objet date en python et j'ai besoin de générer un horodatage dans la locale C pour un système legacy, en utilisant les codes %a (weekday) et %b (month). Toutefois, Je ne souhaite pas modifier la locale de l'application, puisque d'autres parties doivent respecter la locale actuelle de l'utilisateur. Y a-t-il un moyen d'appeler strftime() avec une certaine localisation?

22
demandé sur MagerValp 2013-09-03 17:31:06

3 réponses

L'exemple donné par Rob est grand, mais n'est pas threadsafe. Voici une version qui fonctionne avec les threads:

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

il crée un gestionnaire de contexte threadsafe en utilisant une serrure globale et vous permet d'avoir plusieurs threads en cours d'exécution du code dépendant de la locale en utilisant LOCALE_LOCK. Il traite également les exceptions à l'énoncé de rendement pour s'assurer que la locale originale est toujours restaurée.

25
répondu Daniel 2014-06-05 21:41:50

Non, il n'y a aucun moyen d'appeler

en supposant que votre application n'est pas Multi-threadée, sauvegardez et restaurez la locale existante, et définissez votre locale à 'C' quand vous invoquez 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'))

Si vous préférez with:try: -finally:, vous pouvez concocter un gestionnaire de contexte:

#! /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'))
12
répondu Robᵩ 2013-09-03 14:13:30

jetez un coup d'oeil à la paquet pytz

vous pouvez utiliser comme ceci

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 rendra dans le fuseau horaire spécifié

pour avoir le nom du mois dans la locale utilisez calendrier par exemple :

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

inspectez plus profondément le calendrier pour avoir plus d'informations

-3
répondu Philippe T. 2013-09-03 13:55:18