Computing Standard Deviation in a stream

后端 未结 3 713
梦毁少年i
梦毁少年i 2020-12-01 00:52

Using Python, assume I\'m running through a known quantity of items I, and have the ability to time how long it takes to process each one t, as wel

相关标签:
3条回答
  • 2020-12-01 01:27

    I use Welford's Method, which gives more accurate results. This link points to John D. Cook's overview. Here's a paragraph from it that summarizes why it is a preferred approach:

    This better way of computing variance goes back to a 1962 paper by B. P. Welford and is presented in Donald Knuth’s Art of Computer Programming, Vol 2, page 232, 3rd edition. Although this solution has been known for decades, not enough people know about it. Most people are probably unaware that computing sample variance can be difficult until the first time they compute a standard deviation and get an exception for taking the square root of a negative number.

    0 讨论(0)
  • 2020-12-01 01:30

    As outlined in the Wikipedia article on the standard deviation, it is enough to keep track of the following three sums:

    s0 = sum(1 for x in samples)
    s1 = sum(x for x in samples)
    s2 = sum(x*x for x in samples)
    

    These sums are easily updated as new values arrive. The standard deviation can be calculated as

    std_dev = math.sqrt((s0 * s2 - s1 * s1)/(s0 * (s0 - 1)))
    

    Note that this way of computing the standard deviation can be numerically ill-conditioned if your samples are floating point numbers and the standard deviation is small compared to the mean of the samples. If you expect samples of this type, you should resort to Welford's method (see the accepted answer).

    0 讨论(0)
  • 2020-12-01 01:30

    Based on Welford's algorithm:

    import numpy as np
    
    class OnlineVariance(object):
        """
        Welford's algorithm computes the sample variance incrementally.
        """
    
        def __init__(self, iterable=None, ddof=1):
            self.ddof, self.n, self.mean, self.M2 = ddof, 0, 0.0, 0.0
            if iterable is not None:
                for datum in iterable:
                    self.include(datum)
    
        def include(self, datum):
            self.n += 1
            self.delta = datum - self.mean
            self.mean += self.delta / self.n
            self.M2 += self.delta * (datum - self.mean)
    
        @property
        def variance(self):
            return self.M2 / (self.n - self.ddof)
    
        @property
        def std(self):
            return np.sqrt(self.variance)
    

    Update the variance with each new piece of data:

    N = 100
    data = np.random.random(N)
    ov = OnlineVariance(ddof=0)
    for d in data:
        ov.include(d)
    std = ov.std
    print(std)
    

    Check our result against the standard deviation computed by numpy:

    assert np.allclose(std, data.std())
    
    0 讨论(0)
提交回复
热议问题