Как я могу выбрать все воскресенье в течение года с помощью Python?
Использование Python...
Как я могу выбрать все воскресенье (или любой день, если на то пошло) через год?
[ '01/03/2010','01/10/2010','01/17/2010','01/24/2010', ...]
Эти даты означают воскресенья за 2010 год. Это может также относиться к любому дню недели, я полагаю.
Ответы
Ответ 1
Вы можете использовать date
из модуля datetime
чтобы найти первое воскресенье в году, а затем продолжать добавлять семь дней, генерируя новые воскресенья:
from datetime import date, timedelta
def allsundays(year):
d = date(year, 1, 1) # January 1st
d += timedelta(days = 6 - d.weekday()) # First Sunday
while d.year == year:
yield d
d += timedelta(days = 7)
for d in allsundays(2010):
print(d)
Ответ 2
Используя модуль dateutil, вы можете сгенерировать список следующим образом:
#!/usr/bin/env python
import dateutil.relativedelta as relativedelta
import dateutil.rrule as rrule
import datetime
year=2010
before=datetime.datetime(year,1,1)
after=datetime.datetime(year,12,31)
rr = rrule.rrule(rrule.WEEKLY,byweekday=relativedelta.SU,dtstart=before)
print rr.between(before,after,inc=True)
Несмотря на то, что найти все воскресенья не так сложно, чтобы сделать без даты, модуль удобен, особенно если у вас более сложные или разные расчеты даты.
Если вы используете Debian/Ubuntu, dateutil предоставляется пакетом python-dateutil.
Ответ 3
Вы можете перебирать календарь за этот год.
Нижеследующее должно возвращать все вторники и четверги в течение определенного года.
# Returns all Tuesdays and Thursdays of a given year
from datetime import date
import calendar
year = 2016
c = calendar.TextCalendar(calendar.SUNDAY)
for m in range(1,13):
for i in c.itermonthdays(year,m):
if i != 0: #calendar constructs months with leading zeros (days belongng to the previous month)
day = date(year,m,i)
if day.weekday() == 1 or day.weekday() == 3: #if its Tuesday or Thursday
print "%s-%s-%s" % (year,m,i)
Ответ 4
Pandas имеет большую функциональность для этой цели с помощью date_range
.
Результат: pandas DatetimeIndex
, но может быть легко преобразован в список.
import pandas as pd
def allsundays(year):
return pd.date_range(start=str(year), end=str(year+1),
freq='W-SUN').strftime('%m/%d/%Y').tolist()
allsundays(2017)[:5] # First 5 Sundays of 2017
# ['01/01/2017', '01/08/2017', '01/15/2017', '01/22/2017', '01/29/2017']
Ответ 5
используя понимание списка:
from datetime import date, timedelta
dstart = date(2018,1,1)
dend = date(2018,12,31)
# this will return all sundays between start-end dates.
days = [dstart + timedelta(days=x) for x in range((dend-dstart).days + 1) if (dstart + timedelta(days=x)).weekday() == 6]
Ответ 6
Здесь полная функция генератора, основанная на решении из @sth. Это включает решающее исправление, которое было упомянуто в его комментариях к решению.
Вы можете указать день недели (используя индексацию Python с 0 = понедельник до 6 = воскресенье), дату начала и количество недель для перечисления.
def get_all_dates_of_day_of_week_in_year(day_of_week, start_year, start_month,
start_day, max_weeks=None):
'''
Generator function to enumerate all calendar dates for a specific day
of the week during one year. For example, all Wednesdays in 2018 are:
1/3/2018, 1/10/2018, 1/17/2018, 1/24/2018, 1/31/2018, 2/7/2018, etc.
Parameters:
----------
day_of_week : int
The day_of_week should be one of these values: 0=Monday, 1=Tuesday,
2=Wednesday, 3=Thursday, 4=Friday, 5=Saturday, 6=Sunday.
start_year : int
start_month : int
start_day : int
The starting date from which to list out all the dates
max_weeks : int or None
If None, then list out all dates for the rest of the year.
Otherwise, end the list after max_weeks number of weeks.
'''
if day_of_week < 0 or day_of_week > 6:
raise ValueError('day_of_week should be in [0, 6]')
date_iter = date(start_year, start_month, start_day)
# First desired day_of_week
date_iter += timedelta(days=(day_of_week - date_iter.weekday() + 7) % 7)
week = 1
while date_iter.year == start_year:
yield date_iter
date_iter += timedelta(days=7)
if max_weeks is not None:
week += 1
if week > max_weeks:
break
Пример использования для получения всех сред, начиная с 1 января 2018 года, в течение 10 недель.
import calendar
day_of_week = 2
max_weeks = 10
for d in get_all_dates_of_day_of_week_in_year (day_of_week, 2018, 1, 1, max_weeks):
print "%s, %d/%d/%d" % (calendar.day_name[d.weekday()], d.year, d.month, d.day)
Приведенный выше код производит:
Wednesday, 2018/1/3
Wednesday, 2018/1/10
Wednesday, 2018/1/17
Wednesday, 2018/1/24
Wednesday, 2018/1/31
Wednesday, 2018/2/7
Wednesday, 2018/2/14
Wednesday, 2018/2/21
Wednesday, 2018/2/28
Wednesday, 2018/3/7