Source code for spherical_kde.distributions

""" Module containing the kernel function for the spherical KDE.

For more detail, see:
https://en.wikipedia.org/wiki/Von_Mises-Fisher_distribution
"""

import numpy
import scipy.optimize
from spherical_kde.utils import (cartesian_from_polar,
                                 polar_from_cartesian, logsinh,
                                 rotation_matrix)


[docs]def VonMisesFisher_distribution(phi, theta, phi0, theta0, sigma0): """ Von-Mises Fisher distribution function. Parameters ---------- phi, theta : float or array_like Spherical-polar coordinates to evaluate function at. phi0, theta0 : float or array-like Spherical-polar coordinates of the center of the distribution. sigma0 : float Width of the distribution. Returns ------- float or array_like log-probability of the vonmises fisher distribution. Notes ----- Wikipedia: https://en.wikipedia.org/wiki/Von_Mises-Fisher_distribution """ x = cartesian_from_polar(phi, theta) x0 = cartesian_from_polar(phi0, theta0) norm = -numpy.log(4*numpy.pi*sigma0**2) - logsinh(1./sigma0**2) return norm + numpy.tensordot(x, x0, axes=[[0], [0]])/sigma0**2
[docs]def VonMisesFisher_sample(phi0, theta0, sigma0, size=None): """ Draw a sample from the Von-Mises Fisher distribution. Parameters ---------- phi0, theta0 : float or array-like Spherical-polar coordinates of the center of the distribution. sigma0 : float Width of the distribution. size : int, tuple, array-like number of samples to draw. Returns ------- phi, theta : float or array_like Spherical-polar coordinates of sample from distribution. """ n0 = cartesian_from_polar(phi0, theta0) M = rotation_matrix([0, 0, 1], n0) x = numpy.random.uniform(size=size) phi = numpy.random.uniform(size=size) * 2*numpy.pi theta = numpy.arccos(1 + sigma0**2 * numpy.log(1 + (numpy.exp(-2/sigma0**2)-1) * x)) n = cartesian_from_polar(phi, theta) x = M.dot(n) phi, theta = polar_from_cartesian(x) return phi, theta
[docs]def VonMises_mean(phi, theta): """ Von-Mises sample mean. Parameters ---------- phi, theta : array-like Spherical-polar coordinate samples to compute mean from. Returns ------- float ..math:: \sum_i^N x_i / || \sum_i^N x_i || Notes ----- Wikipedia: https://en.wikipedia.org/wiki/Von_Mises-Fisher_distribution#Estimation_of_parameters """ x = cartesian_from_polar(phi, theta) S = numpy.sum(x, axis=-1) phi, theta = polar_from_cartesian(S) return phi, theta
[docs]def VonMises_std(phi, theta): """ Von-Mises sample standard deviation. Parameters ---------- phi, theta : array-like Spherical-polar coordinate samples to compute mean from. Returns ------- solution for ..math:: 1/tanh(x) - 1/x = R, where ..math:: R = || \sum_i^N x_i || / N Notes ----- Wikipedia: https://en.wikipedia.org/wiki/Von_Mises-Fisher_distribution#Estimation_of_parameters but re-parameterised for sigma rather than kappa. """ x = cartesian_from_polar(phi, theta) S = numpy.sum(x, axis=-1) R = S.dot(S)**0.5/x.shape[-1] def f(s): return 1/numpy.tanh(s)-1./s-R kappa = scipy.optimize.brentq(f, 1e-8, 1e8) sigma = kappa**-0.5 return sigma