Creating a range of dates in Python

2019-01-01 08:13发布

问题:

I want to create a list of dates, starting with today, and going back an arbitrary number of days, say, in my example 100 days. Is there a better way to do it than this?

import datetime

a = datetime.datetime.today()
numdays = 100
dateList = []
for x in range (0, numdays):
    dateList.append(a - datetime.timedelta(days = x))
print dateList

回答1:

Marginally better...

base = datetime.datetime.today()
date_list = [base - datetime.timedelta(days=x) for x in range(0, numdays)]


回答2:

Pandas is great for time series in general, and has direct support for date ranges.

For example pd.date_range():

import pandas as pd
datelist = pd.date_range(pd.datetime.today(), periods=100).tolist()

It also has lots of options to make life easier. For example if you only wanted weekdays, you would just swap in bdate_range.

See http://pandas.pydata.org/pandas-docs/stable/timeseries.html#generating-ranges-of-timestamps

In addition it fully supports pytz timezones and can smoothly span spring/autumn DST shifts.



回答3:

Get range of dates between specified start and end date (Optimized for time & space complexity):

import datetime

start = datetime.datetime.strptime(\"21-06-2014\", \"%d-%m-%Y\")
end = datetime.datetime.strptime(\"07-07-2014\", \"%d-%m-%Y\")
date_generated = [start + datetime.timedelta(days=x) for x in range(0, (end-start).days)]

for date in date_generated:
    print date.strftime(\"%d-%m-%Y\")


回答4:

You can write a generator function that returns date objects starting from today:

import datetime

def date_generator():
  from_date = datetime.datetime.today()
  while True:
    yield from_date
    from_date = from_date - datetime.timedelta(days=1)

This generator returns dates starting from today and going backwards one day at a time. Here is how to take the first 3 dates:

>>> import itertools
>>> dates = itertools.islice(date_generator(), 3)
>>> list(dates)
[datetime.datetime(2009, 6, 14, 19, 12, 21, 703890), datetime.datetime(2009, 6, 13, 19, 12, 21, 703890), datetime.datetime(2009, 6, 12, 19, 12, 21, 703890)]

The advantage of this approach over a loop or list comprehension is that you can go back as many times as you want.

Edit

A more compact version using a generator expression instead of a function:

date_generator = (datetime.datetime.today() - datetime.timedelta(days=i) for i in itertools.count())

Usage:

>>> dates = itertools.islice(date_generator, 3)
>>> list(dates)
[datetime.datetime(2009, 6, 15, 1, 32, 37, 286765), datetime.datetime(2009, 6, 14, 1, 32, 37, 286836), datetime.datetime(2009, 6, 13, 1, 32, 37, 286859)]


回答5:

You can also use the day ordinal to make it simpler:

def date_range(start_date, end_date):
    for ordinal in range(start_date.toordinal(), end_date.toordinal()):
        yield datetime.date.fromordinal(ordinal)

Or as suggested in the comments you can create a list like this:

date_range = [
    datetime.date.fromordinal(ordinal) 
    for ordinal in range(
        start_date.toordinal(),
        end_date.toordinal(),
    )
]


回答6:

yeah, reinvent the wheel.... just search the forum and you\'ll get something like this:

from dateutil import rrule
from datetime import datetime

list(rrule.rrule(rrule.DAILY,count=100,dtstart=datetime.now()))


回答7:

From the title of this question I was expecting to find something like range(), that would let me specify two dates and create a list with all the dates in between. That way one does not need to calculate the number of days between those two dates, if one does not know it beforehand.

So with the risk of being slightly off-topic, this one-liner does the job:

import datetime
start_date = datetime.date(2011, 01, 01)
end_date   = datetime.date(2014, 01, 01)

dates_2011_2013 = [ start_date + datetime.timedelta(n) for n in range(int ((end_date - start_date).days))]

All credits to this answer!



回答8:

A bit of a late answer I know, but I just had the same problem and decided that Python\'s internal range function was a bit lacking in this respect so I\'ve overridden it in a util module of mine.

from __builtin__ import range as _range
from datetime import datetime, timedelta

def range(*args):
    if len(args) != 3:
        return _range(*args)
    start, stop, step = args
    if start < stop:
        cmp = lambda a, b: a < b
        inc = lambda a: a + step
    else:
        cmp = lambda a, b: a > b
        inc = lambda a: a - step
    output = [start]
    while cmp(start, stop):
        start = inc(start)
        output.append(start)

    return output

print range(datetime(2011, 5, 1), datetime(2011, 10, 1), timedelta(days=30))


回答9:

Here is gist I created, from my own code, this might help. (I know the question is too old, but others can use it)

https://gist.github.com/2287345

(same thing below)

import datetime
from time import mktime

def convert_date_to_datetime(date_object):
    date_tuple = date_object.timetuple()
    date_timestamp = mktime(date_tuple)
    return datetime.datetime.fromtimestamp(date_timestamp)

def date_range(how_many=7):
    for x in range(0, how_many):
        some_date = datetime.datetime.today() - datetime.timedelta(days=x)
        some_datetime = convert_date_to_datetime(some_date.date())
        yield some_datetime

def pick_two_dates(how_many=7):
    a = b = convert_date_to_datetime(datetime.datetime.now().date())
    for each_date in date_range(how_many):
        b = a
        a = each_date
        if a == b:
            continue
        yield b, a


回答10:

Here\'s a one liner for bash scripts to get a list of weekdays, this is python 3. Easily modified for whatever, the int at the end is the number of days in the past you want.

python -c \"import sys,datetime; print(\'\\n\'.join([(datetime.datetime.today() - datetime.timedelta(days=x)).strftime(\\\"%Y/%m/%d\\\") for x in range(0,int(sys.argv[1])) if (datetime.datetime.today() - datetime.timedelta(days=x)).isoweekday()<6]))\" 10

Here is a variant to provide a start (or rather, end) date

python -c \"import sys,datetime; print(\'\\n\'.join([(datetime.datetime.strptime(sys.argv[1],\\\"%Y/%m/%d\\\") - datetime.timedelta(days=x)).strftime(\\\"%Y/%m/%d \\\") for x in range(0,int(sys.argv[2])) if (datetime.datetime.today() - datetime.timedelta(days=x)).isoweekday()<6]))\" 2015/12/30 10

Here is a variant for arbitrary start and end dates. not that this isn\'t terribly efficient, but is good for putting in a for loop in a bash script:

python -c \"import sys,datetime; print(\'\\n\'.join([(datetime.datetime.strptime(sys.argv[1],\\\"%Y/%m/%d\\\") + datetime.timedelta(days=x)).strftime(\\\"%Y/%m/%d\\\") for x in range(0,int((datetime.datetime.strptime(sys.argv[2], \\\"%Y/%m/%d\\\") - datetime.datetime.strptime(sys.argv[1], \\\"%Y/%m/%d\\\")).days)) if (datetime.datetime.strptime(sys.argv[1], \\\"%Y/%m/%d\\\") + datetime.timedelta(days=x)).isoweekday()<6]))\" 2015/12/15 2015/12/30


回答11:

Here\'s a slightly different answer building off of S.Lott\'s answer that gives a list of dates between two dates start and end. In the example below, from the start of 2017 to today.

start = datetime.datetime(2017,1,1)
end = datetime.datetime.today()
daterange = [start + datetime.timedelta(days=x) for x in range(0, (end-start).days)]


回答12:

Matplotlib related

from matplotlib.dates import drange
import datetime

base = datetime.date.today()
end  = base + datetime.timedelta(days=100)
delta = datetime.timedelta(days=1)
l = drange(base, end, delta)


回答13:

If there are two dates and you need the range try

from dateutil import rrule, parser
date1 = \'1995-01-01\'
date2 = \'1995-02-28\'
datesx = list(rrule.rrule(rrule.DAILY, dtstart=parser.parse(date1), until=parser.parse(date2)))


回答14:

Based on answers I wrote for myself this:

import datetime;
print [(datetime.date.today() - datetime.timedelta(days=x)).strftime(\'%Y-%m-%d\') for x in range(-5, 0)]

Output:

[\'2017-12-11\', \'2017-12-10\', \'2017-12-09\', \'2017-12-08\', \'2017-12-07\']

The difference is that I get the \'date\' object, not the \'datetime.datetime\' one.



回答15:

I know this has been answered, but I\'ll put down my answer for historical purposes, and since I think it is straight forward.

import numpy as np
import datetime as dt
listOfDates=[date for date in np.arange(firstDate,lastDate,dt.timedelta(days=x))]

Sure it won\'t win anything like code-golf, but I think it is elegant.



回答16:

from datetime import datetime, timedelta
from dateutil import parser
def getDateRange(begin, end):
    \"\"\"  \"\"\"
    beginDate = parser.parse(begin)
    endDate =  parser.parse(end)
    delta = endDate-beginDate
    numdays = delta.days + 1
    dayList = [datetime.strftime(beginDate + timedelta(days=x), \'%Y%m%d\') for x in range(0, numdays)]
    return dayList


回答17:

A monthly date range generator with datetime and dateutil. Simple and easy to understand:

import datetime as dt
from dateutil.relativedelta import relativedelta

def month_range(start_date, n_months):
        for m in range(n_months):
            yield start_date + relativedelta(months=+m)


回答18:

import datetime    
def date_generator():
    cur = base = datetime.date.today()
    end  = base + datetime.timedelta(days=100)
    delta = datetime.timedelta(days=1)
    while(end>base):
        base = base+delta
        print base

date_generator()