Eu tenho o seguinte código para fazer isso, mas como posso fazer isso melhor? No momento, acho que é melhor do que loops aninhados, mas ele começa a ficar com o Perl-one-linerish quando você tem um gerador em uma lista de compreensão.
day_count = (end_date - start_date).days + 1
for single_date in [d for d in (start_date + timedelta(n) for n in range(day_count)) if d <= end_date]:
print strftime("%Y-%m-%d", single_date.timetuple())
Notas
- Na verdade, não estou usando isso para imprimir. Isso é apenas para fins de demonstração.
- As variáveis
start_date
eend_date
sãodatetime.date
objetos porque não preciso dos carimbos de data e hora. (Eles serão usados para gerar um relatório).
Saída de amostra
Para uma data de início 2009-05-30
e uma data final de 2009-06-09
:
2009-05-30
2009-05-31
2009-06-01
2009-06-02
2009-06-03
2009-06-04
2009-06-05
2009-06-06
2009-06-07
2009-06-08
2009-06-09
Respostas:
Por que existem duas iterações aninhadas? Para mim, ele produz a mesma lista de dados com apenas uma iteração:
E nenhuma lista é armazenada, apenas um gerador é iterado. Também o "se" no gerador parece ser desnecessário.
Afinal, uma sequência linear deve exigir apenas um iterador, não dois.
Atualização após discussão com John Machin:
Talvez a solução mais elegante seja usar uma função de gerador para ocultar / abstrair completamente a iteração no intervalo de datas:
Nota: para consistência com a
range()
função incorporada, esta iteração pára antes de atingir oend_date
. Portanto, para iteração inclusiva, use no dia seguinte, como faria comrange()
.fonte
(start_date + datetime.timedelta(n) for n in range((end_date - start_date).days))
Isso pode ser mais claro:
fonte
Use a
dateutil
biblioteca:Essa biblioteca python possui muitos recursos mais avançados, alguns muito úteis, como
relative delta
s - e é implementada como um único arquivo (módulo) que é facilmente incluído em um projeto.fonte
until
que a data final dodaterange
método na resposta de Ber é exclusivo deend_date
.O Pandas é ótimo para séries temporais em geral e tem suporte direto para períodos.
Em seguida, você pode percorrer a faixa de data para imprimir a data:
Ele também tem muitas opções para facilitar a vida. Por exemplo, se você quisesse apenas dias da semana, trocaria bdate_range. Consulte http://pandas.pydata.org/pandas-docs/stable/timeseries.html#generating-ranges-of-timestamps
O poder do Pandas é realmente seus quadros de dados, que suportam operações vetorizadas (como numpy) que tornam as operações em grandes quantidades de dados muito rápidas e fáceis.
EDIT: Você também pode pular completamente o loop for e imprimi-lo diretamente, o que é mais fácil e mais eficiente:
fonte
Essa função faz mais do que você exige estritamente, suportando etapas negativas, etc. Desde que você considere sua lógica de intervalo, você não precisa do separado
day_count
e, o mais importante, o código se torna mais fácil de ler quando você chama a função de vários locais.fonte
Esta é a solução mais legível para humanos que consigo pensar.
fonte
Por que nao tentar:
fonte
A
arange
função do Numpy pode ser aplicada a datas:O uso de
astype
é converter denumpy.datetime64
para uma matriz dedatetime.datetime
objetos.fonte
dates = np.arange(d0, d1, dt).astype(datetime.datetime)
Mostrar os últimos n dias a partir de hoje:
Resultado:
fonte
print((datetime.date.today() + datetime.timedelta(i)).isoformat())
print((datetime.date.today() + datetime.timedelta(i)))
sem o .isoformat () fornece exatamente a mesma saída. Preciso do meu script para imprimir YYMMDD. Alguém sabe como fazer isso?d = datetime.date.today() + datetime.timedelta(i); d.strftime("%Y%m%d")
fonte
fonte
Para completar, o Pandas também possui uma
period_range
função para carimbos de data / hora que estão fora dos limites:fonte
Eu tenho um problema semelhante, mas preciso iterar mensalmente, e não diariamente.
Esta é a minha solução
Exemplo 1
Resultado
Exemplo 2
Resultado
fonte
Pode
't* acredito que esta questão já existe há 9 anos, sem que ninguém sugerindo uma função recursiva simples:Resultado:
Edit: * Agora eu posso acreditar - consulte O Python otimiza a recursão da cauda? . Obrigado Tim .
fonte
Você pode gerar uma série de datas entre duas datas usando a biblioteca do pandas de maneira simples e confiável
Você pode alterar a frequência da geração de datas definindo a frequência como D, M, Q, Y (diariamente, mensalmente, trimestralmente, anualmente)
fonte
fonte
Esta função possui alguns recursos extras:
verificação de erro no caso de o final ser mais antigo que o início
fonte
Aqui está o código para uma função geral do período, semelhante à resposta de Ber, mas mais flexível:
fonte
Que tal o seguinte para executar um intervalo incrementado por dias:
Para uma versão genérica:
Observe que .total_seconds () é suportado somente após o python 2.7 Se você está preso a uma versão anterior, pode escrever sua própria função:
fonte
Abordagem ligeiramente diferente para etapas reversíveis armazenando
range
args em uma tupla.fonte
RESULTADO:
fonte