Come arrotondare il minuto di un oggetto datetime


102

I have a datetime object produced using strptime ().

>>> tm
datetime.datetime(2010, 6, 10, 3, 56, 23)

Quello che devo fare è arrotondare il minuto al decimo minuto più vicino. Quello che ho fatto fino a questo punto è stato prendere il valore minuto e usare round () su di esso.

min = round(tm.minute, -1)

Tuttavia, come nell'esempio precedente, fornisce un tempo non valido quando il valore dei minuti è maggiore di 56, ovvero: 3:60

Qual è un modo migliore per farlo? Non datetimesostenere questo?

Risposte:


134

In questo modo il "pavimento" di un datetimeoggetto memorizzato in tm verrà arrotondato al segno di 10 minuti prima tm.

tm = tm - datetime.timedelta(minutes=tm.minute % 10,
                             seconds=tm.second,
                             microseconds=tm.microsecond)

Se desideri un arrotondamento classico al segno di 10 minuti più vicino, procedi come segue:

discard = datetime.timedelta(minutes=tm.minute % 10,
                             seconds=tm.second,
                             microseconds=tm.microsecond)
tm -= discard
if discard >= datetime.timedelta(minutes=5):
    tm += datetime.timedelta(minutes=10)

o questo:

tm += datetime.timedelta(minutes=5)
tm -= datetime.timedelta(minutes=tm.minute % 10,
                         seconds=tm.second,
                         microseconds=tm.microsecond)

94

Funzione generale per arrotondare un datetime in qualsiasi intervallo di tempo in secondi:

def roundTime(dt=None, roundTo=60):
   """Round a datetime object to any time lapse in seconds
   dt : datetime.datetime object, default now.
   roundTo : Closest number of seconds to round to, default 1 minute.
   Author: Thierry Husson 2012 - Use it as you want but don't blame me.
   """
   if dt == None : dt = datetime.datetime.now()
   seconds = (dt.replace(tzinfo=None) - dt.min).seconds
   rounding = (seconds+roundTo/2) // roundTo * roundTo
   return dt + datetime.timedelta(0,rounding-seconds,-dt.microsecond)

Campioni con arrotondamento di 1 ora e arrotondamento di 30 minuti:

print roundTime(datetime.datetime(2012,12,31,23,44,59,1234),roundTo=60*60)
2013-01-01 00:00:00

print roundTime(datetime.datetime(2012,12,31,23,44,59,1234),roundTo=30*60)
2012-12-31 23:30:00

10
Sfortunatamente questo non funziona con datetime compatibile con tz. Uno dovrebbe usare dt.replace(hour=0, minute=0, second=0)invece di dt.min.
skoval00

2
@ skoval00 + druska Modificato seguendo i tuoi consigli per supportare datetime tz-aware. Grazie!
Le Droid

Grazie @ skoval00 - mi ci è voluto un po 'per capire perché la funzione non funzionava con i miei dati
Mike Santos

1
Questo non funziona affatto per me per lunghi periodi. ad esempio roundTime(datetime.datetime(2012,12,31,23,44,59,1234),roundTo=60*60*24*7)vsroundTime(datetime.datetime(2012,12,30,23,44,59,1234),roundTo=60*60*24*7)
CPBL

Vedi questo per capire il problema:datetime.timedelta(100,1,2,3).seconds == 1
CPBL

15

Dalla migliore risposta che ho modificato a una versione adattata utilizzando solo oggetti datetime, questo evita di dover fare la conversione in secondi e rende il codice chiamante più leggibile:

def roundTime(dt=None, dateDelta=datetime.timedelta(minutes=1)):
    """Round a datetime object to a multiple of a timedelta
    dt : datetime.datetime object, default now.
    dateDelta : timedelta object, we round to a multiple of this, default 1 minute.
    Author: Thierry Husson 2012 - Use it as you want but don't blame me.
            Stijn Nevens 2014 - Changed to use only datetime objects as variables
    """
    roundTo = dateDelta.total_seconds()

    if dt == None : dt = datetime.datetime.now()
    seconds = (dt - dt.min).seconds
    # // is a floor division, not a comment on following line:
    rounding = (seconds+roundTo/2) // roundTo * roundTo
    return dt + datetime.timedelta(0,rounding-seconds,-dt.microsecond)

Campioni con arrotondamento di 1 ora e arrotondamento di 15 minuti:

print roundTime(datetime.datetime(2012,12,31,23,44,59),datetime.timedelta(hour=1))
2013-01-01 00:00:00

print roundTime(datetime.datetime(2012,12,31,23,44,49),datetime.timedelta(minutes=15))
2012-12-31 23:30:00

1
Anche non va bene: print roundTime(datetime.datetime(2012,12,20,23,44,49),datetime.timedelta(days=15)) 2012-12-20 00:00:00mentreprint roundTime(datetime.datetime(2012,12,21,23,44,49),datetime.timedelta(days=15)) 2012-12-21 00:00:00
CPBL

3
Follow-up di cui sopra: sto solo sottolineando che non funziona per delta di tempo arbitrari, ad esempio quelli di oltre 1 giorno. Questa domanda riguarda l'arrotondamento dei minuti, quindi è una restrizione appropriata, ma potrebbe essere più chiara nel modo in cui è scritto il codice.
CPBL

15

Ho usato il codice Stijn Nevens (grazie Stijn) e ho un piccolo componente aggiuntivo da condividere. Arrotondamento per eccesso, per difetto e arrotondamento al più vicino.

aggiornamento 2019-03-09 = commento Spinxz incorporato; grazie.

aggiornamento 2019-12-27 = commento Bart incorporato; grazie.

Testato per date_delta di "X ore" o "X minuti" o "X secondi".

import datetime

def round_time(dt=None, date_delta=datetime.timedelta(minutes=1), to='average'):
    """
    Round a datetime object to a multiple of a timedelta
    dt : datetime.datetime object, default now.
    dateDelta : timedelta object, we round to a multiple of this, default 1 minute.
    from:  http://stackoverflow.com/questions/3463930/how-to-round-the-minute-of-a-datetime-object-python
    """
    round_to = date_delta.total_seconds()
    if dt is None:
        dt = datetime.now()
    seconds = (dt - dt.min).seconds

    if seconds % round_to == 0 and dt.microsecond == 0:
        rounding = (seconds + round_to / 2) // round_to * round_to
    else:
        if to == 'up':
            # // is a floor division, not a comment on following line (like in javascript):
            rounding = (seconds + dt.microsecond/1000000 + round_to) // round_to * round_to
        elif to == 'down':
            rounding = seconds // round_to * round_to
        else:
            rounding = (seconds + round_to / 2) // round_to * round_to

    return dt + datetime.timedelta(0, rounding - seconds, - dt.microsecond)

# test data
print(round_time(datetime.datetime(2019,11,1,14,39,00), date_delta=datetime.timedelta(seconds=30), to='up'))
print(round_time(datetime.datetime(2019,11,2,14,39,00,1), date_delta=datetime.timedelta(seconds=30), to='up'))
print(round_time(datetime.datetime(2019,11,3,14,39,00,776980), date_delta=datetime.timedelta(seconds=30), to='up'))
print(round_time(datetime.datetime(2019,11,4,14,39,29,776980), date_delta=datetime.timedelta(seconds=30), to='up'))
print(round_time(datetime.datetime(2018,11,5,14,39,00,776980), date_delta=datetime.timedelta(seconds=30), to='down'))
print(round_time(datetime.datetime(2018,11,6,14,38,59,776980), date_delta=datetime.timedelta(seconds=30), to='down'))
print(round_time(datetime.datetime(2017,11,7,14,39,15), date_delta=datetime.timedelta(seconds=30), to='average'))
print(round_time(datetime.datetime(2017,11,8,14,39,14,999999), date_delta=datetime.timedelta(seconds=30), to='average'))
print(round_time(datetime.datetime(2019,11,9,14,39,14,999999), date_delta=datetime.timedelta(seconds=30), to='up'))
print(round_time(datetime.datetime(2012,12,10,23,44,59,7769),to='average'))
print(round_time(datetime.datetime(2012,12,11,23,44,59,7769),to='up'))
print(round_time(datetime.datetime(2010,12,12,23,44,59,7769),to='down',date_delta=datetime.timedelta(seconds=1)))
print(round_time(datetime.datetime(2011,12,13,23,44,59,7769),to='up',date_delta=datetime.timedelta(seconds=1)))
print(round_time(datetime.datetime(2012,12,14,23,44,59),date_delta=datetime.timedelta(hours=1),to='down'))
print(round_time(datetime.datetime(2012,12,15,23,44,59),date_delta=datetime.timedelta(hours=1),to='up'))
print(round_time(datetime.datetime(2012,12,16,23,44,59),date_delta=datetime.timedelta(hours=1)))
print(round_time(datetime.datetime(2012,12,17,23,00,00),date_delta=datetime.timedelta(hours=1),to='down'))
print(round_time(datetime.datetime(2012,12,18,23,00,00),date_delta=datetime.timedelta(hours=1),to='up'))
print(round_time(datetime.datetime(2012,12,19,23,00,00),date_delta=datetime.timedelta(hours=1)))

Questo mi ha aiutato. Voglio aggiungerlo se lo si utilizza in PySpark, per analizzare la data e l'ora come una stringa piuttosto che come un oggetto data e ora.
Max

4
L'arrotondamento "per eccesso" forse non sta facendo ciò che la maggior parte delle persone si aspetta. Arrotonderai per eccesso alla data_delta successiva anche se dt non necessitasse di arrotondamento: ad esempio 15: 30: 00.000 con round_to = 60 diventerebbe 15: 31: 00.000
spinxz

L' uparrotondamento è comunque impreciso con questa funzione; 2019-11-07 14:39:00.776980con date_deltauguale ad es. 30 sec e to='up'risulta in 2019-11-07 14:39:00.
Bart,

1
Molte grazie!! Sebbene l' uparrotondamento potrebbe non essere un caso d'uso comune, è necessario quando si tratta di applicazioni che iniziano al limite del minuto
Rahul Bharadwaj

5

Pandas ha una funzione datetime round, ma come con la maggior parte delle cose in Pandas deve essere in formato serie.

>>> ts = pd.Series(pd.date_range(Dt(2019,1,1,1,1),Dt(2019,1,1,1,4),periods=8))
>>> print(ts)
0   2019-01-01 01:01:00.000000000
1   2019-01-01 01:01:25.714285714
2   2019-01-01 01:01:51.428571428
3   2019-01-01 01:02:17.142857142
4   2019-01-01 01:02:42.857142857
5   2019-01-01 01:03:08.571428571
6   2019-01-01 01:03:34.285714285
7   2019-01-01 01:04:00.000000000
dtype: datetime64[ns]

>>> ts.dt.round('1min')
0   2019-01-01 01:01:00
1   2019-01-01 01:01:00
2   2019-01-01 01:02:00
3   2019-01-01 01:02:00
4   2019-01-01 01:03:00
5   2019-01-01 01:03:00
6   2019-01-01 01:04:00
7   2019-01-01 01:04:00
dtype: datetime64[ns]

Documenti : modifica la stringa di frequenza secondo necessità.


Per riferimento, Timestampcontiene floore ceilpure
poulter7

3

se non vuoi usare condition, puoi usare l' modulooperatore:

minutes = int(round(tm.minute, -1)) % 60

AGGIORNARE

volevi qualcosa di simile?

def timeround10(dt):
    a, b = divmod(round(dt.minute, -1), 60)
    return '%i:%02i' % ((dt.hour + a) % 24, b)

timeround10(datetime.datetime(2010, 1, 1, 0, 56, 0)) # 0:56
# -> 1:00

timeround10(datetime.datetime(2010, 1, 1, 23, 56, 0)) # 23:56
# -> 0:00

.. se vuoi il risultato come stringa. per ottenere il risultato datetime, è meglio usare timedelta - vedere altre risposte;)


Ah ma allora il problema qui è che anche l'ora deve aumentare
Lucas Manco

1
@ Lucas Manco - Anche la mia soluzione funziona bene e penso abbia più senso.
Omnifarious

2

lo sto usando. ha il vantaggio di lavorare con datetimes consapevoli di tz.

def round_minutes(some_datetime: datetime, step: int):
    """ round up to nearest step-minutes """
    if step > 60:
        raise AttrbuteError("step must be less than 60")

    change = timedelta(
        minutes= some_datetime.minute % step,
        seconds=some_datetime.second,
        microseconds=some_datetime.microsecond
    )

    if change > timedelta():
        change -= timedelta(minutes=step)

    return some_datetime - change

ha lo svantaggio di lavorare solo per timeslices inferiori a un'ora.


2

Ecco una soluzione generalizzata più semplice senza problemi di precisione in virgola mobile e dipendenze da librerie esterne:

import datetime as dt

def time_mod(time, delta, epoch=None):
    if epoch is None:
        epoch = dt.datetime(1970, 1, 1, tzinfo=time.tzinfo)
    return (time - epoch) % delta

def time_round(time, delta, epoch=None):
    mod = time_mod(time, delta, epoch)
    if mod < (delta / 2):
       return time - mod
    return time + (delta - mod)

Nel tuo caso:

>>> tm
datetime.datetime(2010, 6, 10, 3, 56, 23)
>>> time_round(tm, dt.timedelta(minutes=10))
datetime.datetime(2010, 6, 10, 4, 0)

0
def get_rounded_datetime(self, dt, freq, nearest_type='inf'):

    if freq.lower() == '1h':
        round_to = 3600
    elif freq.lower() == '3h':
        round_to = 3 * 3600
    elif freq.lower() == '6h':
        round_to = 6 * 3600
    else:
        raise NotImplementedError("Freq %s is not handled yet" % freq)

    # // is a floor division, not a comment on following line:
    seconds_from_midnight = dt.hour * 3600 + dt.minute * 60 + dt.second
    if nearest_type == 'inf':
        rounded_sec = int(seconds_from_midnight / round_to) * round_to
    elif nearest_type == 'sup':
        rounded_sec = (int(seconds_from_midnight / round_to) + 1) * round_to
    else:
        raise IllegalArgumentException("nearest_type should be  'inf' or 'sup'")

    dt_midnight = datetime.datetime(dt.year, dt.month, dt.day)

    return dt_midnight + datetime.timedelta(0, rounded_sec)

0

Basato su Stijn Nevens e modificato per uso Django per arrotondare l'ora corrente al minuto 15 più vicino.

from datetime import date, timedelta, datetime, time

    def roundTime(dt=None, dateDelta=timedelta(minutes=1)):

        roundTo = dateDelta.total_seconds()

        if dt == None : dt = datetime.now()
        seconds = (dt - dt.min).seconds
        # // is a floor division, not a comment on following line:
        rounding = (seconds+roundTo/2) // roundTo * roundTo
        return dt + timedelta(0,rounding-seconds,-dt.microsecond)

    dt = roundTime(datetime.now(),timedelta(minutes=15)).strftime('%H:%M:%S')

 dt = 11:45:00

se hai bisogno di data e ora complete, rimuovi il file .strftime('%H:%M:%S')


0

Non è il massimo per la velocità quando viene rilevata l'eccezione, tuttavia funzionerebbe.

def _minute10(dt=datetime.utcnow()):
    try:
        return dt.replace(minute=round(dt.minute, -1))
    except ValueError:
        return dt.replace(minute=0) + timedelta(hours=1)

Tempistiche

%timeit _minute10(datetime(2016, 12, 31, 23, 55))
100000 loops, best of 3: 5.12 µs per loop

%timeit _minute10(datetime(2016, 12, 31, 23, 31))
100000 loops, best of 3: 2.21 µs per loop

0

Una soluzione intuitiva a due righe per arrotondare a una data unità di tempo, qui secondi, per un datetimeoggetto t:

format_str = '%Y-%m-%d %H:%M:%S'
t_rounded = datetime.strptime(datetime.strftime(t, format_str), format_str)

Se desideri arrotondare a un'unità diversa, modifica semplicemente format_str.

Questo approccio non arrotonda a quantità di tempo arbitrarie come i metodi precedenti, ma è un modo piacevolmente pitonico per arrotondare a una data ora, minuto o secondo.


0

Altra soluzione:

def round_time(timestamp=None, lapse=0):
    """
    Round a timestamp to a lapse according to specified minutes

    Usage:

    >>> import datetime, math
    >>> round_time(datetime.datetime(2010, 6, 10, 3, 56, 23), 0)
    datetime.datetime(2010, 6, 10, 3, 56)
    >>> round_time(datetime.datetime(2010, 6, 10, 3, 56, 23), 1)
    datetime.datetime(2010, 6, 10, 3, 57)
    >>> round_time(datetime.datetime(2010, 6, 10, 3, 56, 23), -1)
    datetime.datetime(2010, 6, 10, 3, 55)
    >>> round_time(datetime.datetime(2019, 3, 11, 9, 22, 11), 3)
    datetime.datetime(2019, 3, 11, 9, 24)
    >>> round_time(datetime.datetime(2019, 3, 11, 9, 22, 11), 3*60)
    datetime.datetime(2019, 3, 11, 12, 0)
    >>> round_time(datetime.datetime(2019, 3, 11, 10, 0, 0), 3)
    datetime.datetime(2019, 3, 11, 10, 0)

    :param timestamp: Timestamp to round (default: now)
    :param lapse: Lapse to round in minutes (default: 0)
    """
    t = timestamp or datetime.datetime.now()  # type: Union[datetime, Any]
    surplus = datetime.timedelta(seconds=t.second, microseconds=t.microsecond)
    t -= surplus
    try:
        mod = t.minute % lapse
    except ZeroDivisionError:
        return t
    if mod:  # minutes % lapse != 0
        t += datetime.timedelta(minutes=math.ceil(t.minute / lapse) * lapse - t.minute)
    elif surplus != datetime.timedelta() or lapse < 0:
        t += datetime.timedelta(minutes=(t.minute / lapse + 1) * lapse - t.minute)
    return t

Spero che questo ti aiuti!


0

La via più breve che conosco

min = tm.minute // 10 * 10


0

Quelli sembrano eccessivamente complessi

def round_down_to():
    num = int(datetime.utcnow().replace(second=0, microsecond=0).minute)
    return num - (num%10)

0

Un approccio diretto:

def round_time(dt, round_to_seconds=60):
    """Round a datetime object to any number of seconds
    dt: datetime.datetime object
    round_to_seconds: closest number of seconds for rounding, Default 1 minute.
    """
    rounded_epoch = round(dt.timestamp() / round_to_seconds) * round_to_seconds
    rounded_dt = datetime.datetime.fromtimestamp(rounded_epoch).astimezone(dt.tzinfo)
    return rounded_dt

0

sì, se i tuoi dati appartengono a una colonna DateTime in una serie di panda, puoi arrotondarli per eccesso utilizzando la funzione pandas.Series.dt.round incorporata. Consulta la documentazione qui su pandas.Series.dt.round . Nel tuo caso di arrotondamento a 10 min, sarà Series.dt.round ('10min') o Series.dt.round ('600s') in questo modo:

pandas.Series(tm).dt.round('10min')

Modifica per aggiungere il codice di esempio:

import datetime
import pandas

tm = datetime.datetime(2010, 6, 10, 3, 56, 23)
tm_rounded = pandas.Series(tm).dt.round('10min')
print(tm_rounded)

>>> 0   2010-06-10 04:00:00
dtype: datetime64[ns]

Puoi mostrare come utilizzare ciò che hai suggerito?
DanielM

Non sono sicuro che questa risposta aggiunga qualcosa di nuovo o utile. C'era già una risposta che spiegava la stessa cosa: stackoverflow.com/a/56010357/7851470
Georgy

sì, grazie per avermelo fatto notare. È un mio errore non includere il codice di esempio nella mia risposta e anche non controllare le risposte di tutte le altre persone. Proverò a migliorare questo aspetto.
Nguyen Bryan
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.