Source code for cosapp.utils.distributions.normal
"""Basic class to define a variable distribution."""
from typing import Optional
import scipy.optimize
import scipy.stats
from .distribution import Distribution
[docs]class Normal(Distribution):
"""A class to define a gaussian distribution.
Parameters
----------
worst : float
The parameter absolute worst value.
best : float
The parameter absolute best value.
pworst : float, optional
The worst value probability is the probability that the variable will be lower (if worst<best) or higher
(if worst>best) than the worst value; default 0.15 (i.e. 15%).
pbest : float, optional
The best value probability is the probability that the variable will be higher (if worst<best) or lower
(if worst>best) than the best value; default 0.15 (i.e. 15%).
"""
def __init__(
self,
worst: float,
best: float,
pworst: Optional[float] = 0.15,
pbest: Optional[float] = 0.15,
):
self._rv = None # type: scipy.stats.norm
super().__init__(worst, best, pworst, pbest)
def _set_distribution(self) -> None:
"""Set the probability distribution according the parameters."""
if self.pworst + self.pbest > 1.0:
raise ValueError(
f"Best and worst probabilities are incompatible: {self.__json__()!s}."
)
pts = [self.worst, self.best]
if self.worst > self.best:
ppts = [(1.0 - self.pworst), self.pbest]
else:
ppts = [self.pworst, (1.0 - self.pbest)]
if self._rv is None:
x0 = [0.0, 1.0]
else:
params = self._rv.kwds # return {"loc": #, "scale": #}
x0 = [params["loc"], params["scale"]]
def f(x):
t = scipy.stats.norm(loc=x[0], scale=x[1])
ppf = t.ppf(ppts)
return ppf - pts
res = scipy.optimize.root(f, x0)
if not res.success:
raise ValueError(
f"Unable to fit normal distribution on {self.__json__()!s}."
)
self._rv = scipy.stats.norm(loc=res.x[0], scale=res.x[1])
[docs] def draw(self, quantile: Optional[float] = None) -> float:
"""Generate a random number.
If a quantile is given, generate the perturbation for that quantile.
Parameters
----------
quantile : Optional[float], optional
Quantile for which the perturbation must be set; default None (i.e. random perturbation)
Returns
-------
float
The random number
"""
return self._rv.rvs() if quantile is None else self._rv.ppf(quantile)