2010-03-09 17 views
52

numpy.average() hat eine Gewichtungsoption, aber numpy.std() nicht. Hat jemand Vorschläge für eine Problemumgehung?Gewichtete Standardabweichung in NumPy?

+0

Btw, Berechnung der gewichteten std dev ist eigentlich ein ziemlich komplexes Thema - es gibt mehr als einen Weg, es zu tun. Sehen Sie hier für eine gute Diskussion: https://www.stata.com/support/faqs/statistics/weights-and-summary-statistics/ – JohnE

Antwort

80

Wie wäre es mit der folgenden kurzen "manuellen Berechnung"?

def weighted_avg_and_std(values, weights): 
    """ 
    Return the weighted average and standard deviation. 

    values, weights -- Numpy ndarrays with the same shape. 
    """ 
    average = numpy.average(values, weights=weights) 
    # Fast and numerically precise: 
    variance = numpy.average((values-average)**2, weights=weights) 
    return (average, math.sqrt(variance)) 
+2

Warum nicht 'numpy.average' noch einmal für die Varianz verwenden? – user2357112

+4

Ich wollte nur darauf hinweisen, dass dies die verzerrte Varianz ergibt. Bei kleinen Stichprobengrößen sollten Sie die Varianz (vor sqrt) neu skalieren, um die unvoreingenommene Varianz zu erhalten. Siehe https://en.wikipedia.org/wiki/Weighted_variance#Weighted_sample_variance – Corey

+1

Ja, der unvoreingenommene Varianzschätzer würde etwas anders sein. Diese Antwort gibt die Standardabweichung an, da die Frage nach einer gewichteten Version von 'numpy.std()' fragt. – EOL

6

Es scheint eine solche Funktion in numpy/scipy noch nicht zu sein, aber es gibt eine ticket, die diese zusätzliche Funktionalität vorschlägt. Dort finden Sie Statistics.py, die gewichtete Standardabweichungen implementiert.

13

Es gibt eine Klasse gewichtete Statistik in statsmodels ist zu berechnen: statsmodels.stats.weightstats.DescrStatsW:

from statsmodels.stats.weightstats import DescrStatsW 

array = np.array([1,2,1,2,1,2,1,3]) 
weights = np.ones_like(array) 
weights[3] = 100 

weighted_stats = DescrStatsW(array, weights=weights, ddof=0) 

weighted_stats.mean  # weighted mean of data (equivalent to np.average(array, weights=weights)) 
# 1.97196261682243 

weighted_stats.std  # standard deviation with default degrees of freedom correction 
# 0.21434289609681711 

weighted_stats.std_mean # standard deviation of weighted mean 
# 0.020818822467555047 

weighted_stats.var  # variance with default degrees of freedom correction 
# 0.045942877107170932 

Die nette Eigenschaft dieser Klasse ist, dass, wenn Sie verschiedene statistische Eigenschaften nachfolgende Anrufe sehr schnell sein wird, berechnet werden soll, weil Bereits berechnete (auch Zwischen-) Ergebnisse werden zwischengespeichert.

1

Es ist ein sehr gutes Beispiel vorgeschlagen von gaborous:

import pandas as pd 
import numpy as np 
# X is the dataset, as a Pandas' DataFrame 
mean = mean = np.ma.average(X, axis=0, weights=weights) # Computing the 
weighted sample mean (fast, efficient and precise) 

# Convert to a Pandas' Series (it's just aesthetic and more 
# ergonomic; no difference in computed values) 
mean = pd.Series(mean, index=list(X.keys())) 
xm = X-mean # xm = X diff to mean 
xm = xm.fillna(0) # fill NaN with 0 (because anyway a variance of 0 is 
just void, but at least it keeps the other covariance's values computed 
correctly)) 
sigma2 = 1./(w.sum()-1) * xm.mul(w, axis=0).T.dot(xm); # Compute the 
unbiased weighted sample covariance 

Correct equation for weighted unbiased sample covariance, URL (version: 2016-06-28)