Source code for preliz.distributions.skewnormal

import numpy as np
from pytensor_distributions import skewnormal as ptd_skewnormal
from scipy.stats import skew

from preliz.distributions.distributions import Continuous
from preliz.internal.distribution_helper import (
    all_not_none,
    eps,
    from_precision,
    pytensor_jit,
    pytensor_rng_jit,
    to_precision,
)
from preliz.internal.optimization import optimize_mean_sigma, optimize_ml


[docs] class SkewNormal(Continuous): r""" SkewNormal distribution. The pdf of this distribution is .. math:: f(x \mid \mu, \tau, \alpha) = 2 \Phi((x-\mu)\sqrt{\tau}\alpha) \phi(x,\mu,\tau) .. plot:: :context: close-figs from preliz import SkewNormal, style style.use('preliz-doc') for alpha in [-6, 0, 6]: SkewNormal(mu=0, sigma=1, alpha=alpha).plot_pdf() ======== ========================================== Support :math:`x \in \mathbb{R}` Mean :math:`\mu + \sigma \sqrt{\frac{2}{\pi}} \frac {\alpha }{{\sqrt {1+\alpha ^{2}}}}` Variance :math:`\sigma^2 \left( 1-\frac{2\alpha^2}{(\alpha^2+1) \pi} \right)` ======== ========================================== SkewNormal distribution has 2 alternative parameterizations. In terms of mu, sigma (standard deviation) and alpha, or mu, tau (precision) and alpha. The link between the 2 alternatives is given by .. math:: \tau = \frac{1}{\sigma^2} Parameters ---------- mu : float Location parameter. sigma : float Scale parameter (sigma > 0). alpha : float Skewness parameter. tau : float Precision (tau > 0). Notes ----- When alpha=0 we recover the Normal distribution and mu becomes the mean, and sigma the standard deviation. In the limit of alpha approaching plus/minus infinite we get a half-normal distribution. """ def __init__(self, mu=None, sigma=None, alpha=None, tau=None): super().__init__() self.support = (-np.inf, np.inf) self._parametrization(mu, sigma, alpha, tau) def _parametrization(self, mu=None, sigma=None, alpha=None, tau=None): if all_not_none(sigma, tau): raise ValueError( "Incompatible parametrization. Either use mu, sigma and alpha," " or mu, tau and alpha." ) self.param_names = ("mu", "sigma", "alpha") self.params_support = ((-np.inf, np.inf), (eps, np.inf), (-np.inf, np.inf)) if tau is not None: self.tau = tau sigma = from_precision(tau) self.param_names = ("mu", "tau", "alpha") self.mu = mu self.sigma = sigma self.alpha = alpha if all_not_none(self.mu, self.sigma, self.alpha): self._update(self.mu, self.sigma, self.alpha) def _update(self, mu, sigma, alpha): self.mu = np.float64(mu) self.sigma = np.float64(sigma) self.alpha = np.float64(alpha) self.tau = to_precision(sigma) if self.param_names[1] == "sigma": self.params = (self.mu, self.sigma, self.alpha) elif self.param_names[1] == "tau": self.params = (self.mu, self.tau, self.alpha) self.is_frozen = True
[docs] def pdf(self, x): return ptd_pdf(x, self.mu, self.sigma, self.alpha)
[docs] def cdf(self, x): return ptd_cdf(x, self.mu, self.sigma, self.alpha)
[docs] def ppf(self, q): return ptd_ppf(q, self.mu, self.sigma, self.alpha)
[docs] def logpdf(self, x): return ptd_logpdf(x, self.mu, self.sigma, self.alpha)
[docs] def entropy(self): return ptd_entropy(self.mu, self.sigma, self.alpha)
[docs] def mean(self): return ptd_mean(self.mu, self.sigma, self.alpha)
[docs] def median(self): return ptd_median(self.mu, self.sigma, self.alpha)
[docs] def var(self): return ptd_var(self.mu, self.sigma, self.alpha)
[docs] def std(self): return ptd_std(self.mu, self.sigma, self.alpha)
[docs] def skewness(self): return ptd_skewness(self.mu, self.sigma, self.alpha)
[docs] def kurtosis(self): return ptd_kurtosis(self.mu, self.sigma, self.alpha)
[docs] def lmoment1(self): return ptd_lmoment1(self.mu, self.sigma, self.alpha)
[docs] def lmoment2(self): return ptd_lmoment2(self.mu, self.sigma, self.alpha)
[docs] def lmoment3(self): return ptd_lmoment3(self.mu, self.sigma, self.alpha)
[docs] def lmoment4(self): return ptd_lmoment4(self.mu, self.sigma, self.alpha)
[docs] def rvs(self, size=None, random_state=None): random_state = np.random.default_rng(random_state) return ptd_rvs(self.mu, self.sigma, self.alpha, size=size, rng=random_state)
def _fit_moments(self, mean, sigma): if self.alpha is None: self.alpha = 0 optimize_mean_sigma(self, mean, sigma) def _fit_mle(self, sample): skewness = skew(sample) self.alpha = skewness / (1 - skewness**2) ** 0.5 optimize_ml(self, sample)
@pytensor_jit def ptd_pdf(x, mu, sigma, alpha): return ptd_skewnormal.pdf(x, mu, sigma, alpha) @pytensor_jit def ptd_cdf(x, mu, sigma, alpha): return ptd_skewnormal.cdf(x, mu, sigma, alpha) @pytensor_jit def ptd_ppf(q, mu, sigma, alpha): return ptd_skewnormal.ppf(q, mu, sigma, alpha) @pytensor_jit def ptd_logpdf(x, mu, sigma, alpha): return ptd_skewnormal.logpdf(x, mu, sigma, alpha) @pytensor_jit def ptd_entropy(mu, sigma, alpha): return ptd_skewnormal.entropy(mu, sigma, alpha) @pytensor_jit def ptd_mean(mu, sigma, alpha): return ptd_skewnormal.mean(mu, sigma, alpha) @pytensor_jit def ptd_mode(mu, sigma, alpha): return ptd_skewnormal.mode(mu, sigma, alpha) @pytensor_jit def ptd_median(mu, sigma, alpha): return ptd_skewnormal.median(mu, sigma, alpha) @pytensor_jit def ptd_var(mu, sigma, alpha): return ptd_skewnormal.var(mu, sigma, alpha) @pytensor_jit def ptd_std(mu, sigma, alpha): return ptd_skewnormal.std(mu, sigma, alpha) @pytensor_jit def ptd_skewness(mu, sigma, alpha): return ptd_skewnormal.skewness(mu, sigma, alpha) @pytensor_jit def ptd_kurtosis(mu, sigma, alpha): return ptd_skewnormal.kurtosis(mu, sigma, alpha) @pytensor_jit def ptd_lmoment1(mu, sigma, alpha): return ptd_skewnormal.lmoment1(mu, sigma, alpha) @pytensor_jit def ptd_lmoment2(mu, sigma, alpha): return ptd_skewnormal.lmoment2(mu, sigma, alpha) @pytensor_jit def ptd_lmoment3(mu, sigma, alpha): return ptd_skewnormal.lmoment3(mu, sigma, alpha) @pytensor_jit def ptd_lmoment4(mu, sigma, alpha): return ptd_skewnormal.lmoment4(mu, sigma, alpha) @pytensor_rng_jit def ptd_rvs(mu, sigma, alpha, size, rng): return ptd_skewnormal.rvs(mu, sigma, alpha, size=size, random_state=rng)