pingouin.circ_r#
- pingouin.circ_r(angles, w=None, d=None, axis=0)[source]#
Mean resultant vector length for circular data.
- Parameters:
- anglesarray_like
Samples of angles in radians. The range of
angles
must be either \([0, 2\pi]\) or \([-\pi, \pi]\). Ifangles
is not expressed in radians (e.g. degrees or 24-hours), please use thepingouin.convert_angles()
function prior to using the present function.- warray_like
Number of incidences per bins (i.e. “weights”), in case of binned angle data.
- dfloat
Spacing (in radians) of bin centers for binned data. If supplied, a correction factor is used to correct for bias in the estimation of r.
- axisint or None
Compute along this dimension. Default is the first axis (0).
- Returns:
- rfloat
Circular mean vector length.
See also
Notes
The length of the mean resultant vector is a crucial quantity for the measurement of circular spread or hypothesis testing in directional statistics. The closer it is to one, the more concentrated the data sample is around the mean direction (Berens 2009).
The circular vector length of a set of angles \(\alpha\) is defined by:
\[\bar{\alpha} = \frac{1}{N}\left \| \sum_{j=1}^n \exp(i \cdot \alpha_j) \right \|\]Missing values in
angles
are omitted from the calculations.References
Berens, P. (2009). CircStat: A MATLAB Toolbox for Circular Statistics. Journal of Statistical Software, Articles, 31(10), 1–21. https://doi.org/10.18637/jss.v031.i10
Examples
Mean resultant vector length of a 1-D array of angles, in radians
>>> import pingouin as pg >>> angles = [0.785, 1.570, 3.141, 0.839, 5.934] >>> r = pg.circ_r(angles) >>> round(r, 4) 0.4972
Note that there is a close relationship between the vector length and the circular standard deviation, i.e. \(\sigma = \sqrt{-2 \ln R}\):
>>> import numpy as np >>> round(np.sqrt(-2 * np.log(r)), 4) 1.1821
which gives similar result as SciPy built-in function:
>>> from scipy.stats import circstd >>> round(circstd(angles), 4) 1.1821
Sanity check: if all angles are the same, the vector length should be one:
>>> angles = [3.14, 3.14, 3.14, 3.14] >>> round(pg.circ_r(angles), 4) 1.0
Using a 2-D array of angles in degrees
>>> np.random.seed(123) >>> deg = np.random.randint(low=0, high=360, size=(3, 5)) >>> deg array([[322, 98, 230, 17, 83], [106, 123, 57, 214, 225], [ 96, 113, 126, 47, 73]])
We first need to convert from degrees to radians:
>>> rad = np.round(pg.convert_angles(deg, low=0, high=360), 4) >>> rad array([[-0.6632, 1.7104, -2.2689, 0.2967, 1.4486], [ 1.85 , 2.1468, 0.9948, -2.5482, -2.3562], [ 1.6755, 1.9722, 2.1991, 0.8203, 1.2741]])
>>> pg.circ_r(rad) # On the first axis (default) array([0.46695499, 0.98398294, 0.3723287 , 0.31103746, 0.42527149]) >>> pg.circ_r(rad, axis=-1) # On the last axis (default) array([0.28099998, 0.45456096, 0.88261161]) >>> round(pg.circ_r(rad, axis=None), 4) # Across the entire array 0.4486
Missing values are omitted from the calculations:
>>> rad[0, 0] = np.nan >>> pg.circ_r(rad) array([0.99619613, 0.98398294, 0.3723287 , 0.31103746, 0.42527149])
Using binned angles
>>> np.random.seed(123) >>> nbins = 18 # Number of bins to divide the unit circle >>> angles_bins = np.linspace(0, 2 * np.pi, nbins) >>> # w represents the number of incidences per bins, or "weights". >>> w = np.random.randint(low=0, high=5, size=angles_bins.size) >>> round(pg.circ_r(angles_bins, w), 4) 0.3642