diff --git a/src/statistics/slice_statistics.rs b/src/statistics/slice_statistics.rs index 3b1c811a..346716f3 100644 --- a/src/statistics/slice_statistics.rs +++ b/src/statistics/slice_statistics.rs @@ -381,18 +381,29 @@ impl + AsRef<[f64]>> StandardizedMoment for Data { // TODO: rewrite iterator statistics and rely on them instead let mut iter = self.0.as_ref().iter().enumerate(); - let mut sum = *iter.next()?.1; - let mut variance = 0.0; + // init single sample (mean=sample and M_2=0) + let mut n: f64 = 1.; + let mut mu = *iter.next()?.1; + let mut m2 = 0.0; for (i, &x) in iter { if x.is_nan() { + // fail early return Some(f64::NAN); } - let diff = (i + 1) as f64 * x - sum; - sum += x; - variance += diff * diff / (i * (i + 1)) as f64 + n = (i + 1) as f64; + let delta = x - mu; + mu += delta / n; + m2 += delta * (x - mu); + } + + if n == 1. { + // no sample variance for a single sample + None + } else { + // normalize by bessel + Some(m2 / (n - 1.)) } - Some(variance / (self.0.as_ref().len() - 1) as f64) } fn skewness(&self) -> Self::Skew {