Comment rééchantillonner une dataframe avec différentes fonctions appliquées à chaque colonne?

j'ai une série de temps avec la température et le rayonnement dans un pandas dataframe . La résolution temporelle est d'une minute par pas réguliers.

import datetime
import pandas as pd
import numpy as np

date_times = pd.date_range(datetime.datetime(2012, 4, 5, 8, 0),
                           datetime.datetime(2012, 4, 5, 12, 0),
                           freq='1min')
tamb = np.random.sample(date_times.size) * 10.0
radiation = np.random.sample(date_times.size) * 10.0
frame = pd.DataFrame(data={'tamb': tamb, 'radiation': radiation},
                     index=date_times)
frame
<class 'pandas.core.frame.DataFrame'>
DatetimeIndex: 241 entries, 2012-04-05 08:00:00 to 2012-04-05 12:00:00
Freq: T
Data columns:
radiation    241  non-null values
tamb         241  non-null values
dtypes: float64(2)

Comment puis-je échantillonner ce dataframe à une résolution d'une heure, en calculant le horaire moyen pour la température et le horaire "somme pour le rayonnement?

24
demandé sur piRSquared 2012-04-05 03:17:23

4 réponses

avec pandas 0.18 l'API de rééchantillonnage a changé (voir le docs ). Ainsi, pour les pandas >= 0,18, la réponse est:

In [31]: frame.resample('1H').agg({'radiation': np.sum, 'tamb': np.mean})
Out[31]: 
                         tamb   radiation
2012-04-05 08:00:00  5.161235  279.507182
2012-04-05 09:00:00  4.968145  290.941073
2012-04-05 10:00:00  4.478531  317.678285
2012-04-05 11:00:00  4.706206  335.258633
2012-04-05 12:00:00  2.457873    8.655838

Vieille Réponse:

je réponds à ma question pour refléter les changements relatifs aux séries chronologiques dans pandas >= 0.8 (toutes les autres réponses sont périmées).

utilisant pandas > = 0.8 la réponse est:

In [30]: frame.resample('1H', how={'radiation': np.sum, 'tamb': np.mean})
Out[30]: 
                         tamb   radiation
2012-04-05 08:00:00  5.161235  279.507182
2012-04-05 09:00:00  4.968145  290.941073
2012-04-05 10:00:00  4.478531  317.678285
2012-04-05 11:00:00  4.706206  335.258633
2012-04-05 12:00:00  2.457873    8.655838
45
répondu bmu 2016-04-08 12:36:22

vous pouvez également descendre l'exemple en utilisant la asof méthode de pandas.DateRange objets .

In [21]: hourly = pd.DateRange(datetime.datetime(2012, 4, 5, 8, 0),
...                          datetime.datetime(2012, 4, 5, 12, 0),
...                          offset=pd.datetools.Hour())

In [22]: frame.groupby(hourly.asof).size()
Out[22]: 
key_0
2012-04-05 08:00:00    60
2012-04-05 09:00:00    60
2012-04-05 10:00:00    60
2012-04-05 11:00:00    60
2012-04-05 12:00:00    1
In [23]: frame.groupby(hourly.asof).agg({'radiation': np.sum, 'tamb': np.mean})
Out[23]: 
                     radiation  tamb 
key_0                                
2012-04-05 08:00:00  271.54     4.491
2012-04-05 09:00:00  266.18     5.253
2012-04-05 10:00:00  292.35     4.959
2012-04-05 11:00:00  283.00     5.489
2012-04-05 12:00:00  0.5414     9.532
3
répondu Garrett 2012-04-05 01:45:17

pour vous séduire, dans pandas 0.8.0 (sous développement lourd dans la branche timeseries sur GitHub), vous pourrez faire:

In [5]: frame.convert('1h', how='mean')
Out[5]: 
                     radiation      tamb
2012-04-05 08:00:00   7.840989  8.446109
2012-04-05 09:00:00   4.898935  5.459221
2012-04-05 10:00:00   5.227741  4.660849
2012-04-05 11:00:00   4.689270  5.321398
2012-04-05 12:00:00   4.956994  5.093980

les méthodes mentionnées ci-dessus sont la bonne stratégie avec la version actuelle de production de pandas.

3
répondu Wes McKinney 2012-04-07 20:34:43

vous devez utiliser groupby comme tel:

grouped = frame.groupby(lambda x: x.hour)
grouped.agg({'radiation': np.sum, 'tamb': np.mean})
# Same as: grouped.agg({'radiation': 'sum', 'tamb': 'mean'})

avec la sortie étant:

        radiation      tamb
key_0                      
8      298.581107  4.883806
9      311.176148  4.983705
10     315.531527  5.343057
11     288.013876  6.022002
12       5.527616  8.507670

donc, essentiellement, je divise sur la valeur de l'heure et puis le calcul de la moyenne de tamb et la somme de radiation et le retour de la DataFrame (approche similaire à R ddply ). Pour plus d'informations je voudrais vérifier la page de documentation pour groupby ainsi que ce blog.

Edit: Pour faire de cette échelle un peu mieux vous pouvez regrouper à la fois le jour et l'heure comme suit:

grouped = frame.groupby(lambda x: (x.day, x.hour))
grouped.agg({'radiation': 'sum', 'tamb': 'mean'})
          radiation      tamb
key_0                        
(5, 8)   298.581107  4.883806
(5, 9)   311.176148  4.983705
(5, 10)  315.531527  5.343057
(5, 11)  288.013876  6.022002
(5, 12)    5.527616  8.507670
0
répondu diliop 2012-04-05 01:24:14