Я искал ответ через stackoverflow, но мне не удалось найти то, что я ищу, в Python и питоническим способом.
Я пытаюсь получить количество дней, недель или месяцев вперед на основе двух дат. Вот небольшой сценарий, который я создал, который делает то, что я хочу делать, но меня это беспокоит.
import datetime
from dateutil.relativedelta import relativedelta
now = datetime.datetime.now()
days_ahead = datetime.datetime.now() + relativedelta(days=3)
weeks_ahead = datetime.datetime.now() + relativedelta(weeks=2)
month_ahead = datetime.datetime.now() + relativedelta(months=1)
months_ahead = datetime.datetime.now() + relativedelta(months=3)
def get_relative_date(dt):
ahead = (dt - now).days
if ahead < 7:
return "Due in " + str(ahead) + " days"
elif ahead < 31:
return "Due in " + str(ahead/7) + " weeks"
else:
return "Due in " + str(ahead/30) + " months"
print get_relative_date(days_ahead)
print get_relative_date(weeks_ahead)
print get_relative_date(month_ahead)
print get_relative_date(months_ahead)
Результат следующий:
Due in 3 days
Due in 2 weeks
Due in 1 months
Due in 3 months
Несмотря на то, что это хороший ответ, мои опасения связаны с:
- Я использую
ahead < 30
, но как насчет месяцев с 31 днем? Не вызовет ли это каких-то накладных расходов и ошибок в какой-то момент? - Есть ли лучший способ получить эту информацию? Какая-то библиотека или встроенная функция для datetime или dateutil, которая возвращает вам эту информацию?
Заранее спасибо. Если был дан ответ на вопрос, пожалуйста, свяжите меня с постом, и я внимательно его прочту. Я готов предоставить дополнительную информацию, если потребуется.
Редактировать
Я включаю сюда свой полный обновленный код для всех, кому тоже нужна эта функциональность в Python. Он также заботится об отрицательных значениях дня и сегодняшнего дня.
def relative_date(dt):
if dt is not None and len(dt) > 0:
now = datetime.now()
then = arrow.get(dt).naive
rd = relativedelta(then, now)
if rd.years or rd.months:
months = 12 * rd.years + rd.months
if months < 0:
if months == -1:
return "Due 1 month ago"
return "Due %i months ago" % -months
if months == 1:
return "Due in 1 month"
return "Due in %d months" % months
elif rd.days > 7 or rd.days < -7:
weeks = rd.days / 7
if weeks < 0:
if weeks == -1:
return "Due 1 week ago"
return "Due %i weeks ago" % -weeks
if weeks == 1:
return "Due in 1 week"
return "Due in %d weeks" % weeks
else:
if rd.days == 0:
return "Due Today"
elif rd.days < 0:
if rd.days == -1:
return "Due 1 day ago"
return "Due %i days ago" % -rd.days
elif rd.days == 1:
return "Due in 1 day"
return "Due in %d days" % rd.days
else:
return ""