SMA
Simple Moving Average — the equal-weighted rolling mean of the last
periodcloses, maintained as an O(1) rolling-sum state machine.
Quick reference
| Field | Value |
|---|---|
| Family | Moving Averages |
| Input type | f64 (single close) |
| Output type | f64 |
| Output range | unbounded; tracks the input price scale |
| Default parameters | period is required (no default in either binding) |
| Warmup period | period |
| Interpretation | Smoothed price level; price-vs-SMA crossings flag direction changes. |
Formula
SMA_t = (1 / n) * Σ_{i=0}^{n-1} price_{t-i}where n = period. Maintained incrementally as sum -= window.pop_front(); sum += new_price; out = sum / n, so update is O(1) regardless of period. To keep f64 rounding error bounded on long-running streams (where catastrophic cancellation between add/subtract pairs could otherwise accumulate), the running sum is reseeded from the live window every 16 · period updates — still amortised O(1) (O(period) work amortised over O(period) updates), zero observable change on inputs that did not drift to begin with.
Parameters
| Name | Type | Default | Valid range | Description |
|---|---|---|---|---|
period | usize | none | >= 1 | Length of the rolling window. period = 0 errors with Error::PeriodZero. period = 1 is a pass-through. |
(There is no Python #[pyo3(signature = …)] default for SMA, so wickra.SMA(period) requires the period explicitly.)
Inputs / Outputs
From crates/wickra-core/src/indicators/sma.rs:
use wickra::{Indicator, Sma};
// Sma: Input = f64, Output = f64
const _: fn(&mut Sma, f64) -> Option<f64> = <Sma as Indicator>::update;A single f64 close in, an Option<f64> out. The Python binding maps this to float | None (streaming) or a numpy.ndarray of dtype float64 with NaN for warmup rows (batch). The Node binding maps it to number | null / Array<number> with NaN for warmup.
Warmup
Sma::new(period).warmup_period() == period. The first non-empty value is emitted on the period-th update() call, because the window needs to hold exactly period values before the mean is defined. There is no seeding step beyond filling the window — Sma only ever stores its running sum and the VecDeque of values, so its readiness condition is literally window.len() == period.
Edge cases
- Constant series. Feeding
[7.0; n]returnsSome(7.0)from inputperiodonward; the running-sum bookkeeping is exact for constants (the unit testconstant_series_yields_constant_smapins this). - NaN / infinity inputs. The first line of
updateisif !input.is_finite() { return self.value(); }. Non-finite inputs are silently dropped — they do not advance the window, do not corrupt the sum, and the previous valid value (if any) is returned. The unit testignores_non_finite_input_but_keeps_statepins this behaviour. - Reset.
sma.reset()clears the window and the sum, returning the indicator to a freshis_ready() == falsestate. The nextupdatestarts a new warmup countdown.
Examples
Rust
use wickra::{BatchExt, Indicator, Sma};
fn main() -> Result<(), Box<dyn std::error::Error>> {
let mut sma = Sma::new(3)?;
let out: Vec<Option<f64>> = sma.batch(&[2.0, 4.0, 6.0, 8.0, 10.0]);
println!("{:?}", out);
println!("warmup_period = {}", sma.warmup_period());
Ok(())
}Output:
[None, None, Some(4.0), Some(6.0), Some(8.0)]
warmup_period = 3The first two inputs return None while the window fills; the third emits (2 + 4 + 6) / 3 = 4.0 and every subsequent input slides the window by one. This matches the known_reference_values test in crates/wickra-core/src/indicators/sma.rs.
Python
import numpy as np
import wickra as ta
sma = ta.SMA(3)
print(sma.batch(np.array([2.0, 4.0, 6.0, 8.0, 10.0])))
print("warmup_period =", sma.warmup_period())Output:
[nan nan 4. 6. 8.]
warmup_period = 3Warmup rows come back as NaN so the result aligns 1:1 with the input array.
Node
const ta = require('wickra');
const sma = new ta.SMA(3);
console.log(sma.batch([2, 4, 6, 8, 10]));
console.log('warmupPeriod:', sma.warmupPeriod());Output:
[ NaN, NaN, 4, 6, 8 ]
warmupPeriod: 3Interpretation
Sma is a smoothed price level. The two canonical signals are:
- Price–SMA crossover. Close above the SMA suggests an uptrend, close below suggests a downtrend. The longer the SMA, the slower (and more trustworthy) the signal.
- Two-SMA crossover. A fast SMA crossing above a slow SMA is the classic "golden cross"; below is the "death cross". Either of
EmaorHmawill give earlier (but noisier) signals at the same period.
Prefer Sma when you want the simplest possible reference price — for example, as the middle band of BollingerBands, which uses an SMA by construction. Prefer Ema if you want the same smoothness profile but slightly less lag on direction changes.
Common pitfalls
- Treating
period = 0as "use a default".Sma::new(0)returnsErr(Error::PeriodZero)in Rust and aValueErrorin Python; there is no implicit default. Pass an explicit period. - Slicing batch results with
> warmup_periodinstead of~np.isnan(...). In Python the batch output hasNaNfor warmup rows; in Rust it hasNone. Use the warmup-aware mask to filter — see the Quickstart: Python pattern. Slicing byprices.size - warmup_periodworks for a single indicator but breaks the moment you compose two of them viaChain.
References
The simple moving average predates technical analysis as a discipline. The implementation here follows the standard "rolling sum, slide on each update" formulation; the matching reference implementations are TA-Lib and pandas (rolling(period).mean()).
See also
- Indicator-Ema — same smoothness budget, less lag.
- Indicator-Wma — linear weights instead of equal.
- Indicator-Hma — built on three WMAs for near-zero lag.
- Indicators-Overview — the full taxonomy.