Source code for mixem.distribution.normal
# coding=utf-8
import numpy as np
import scipy.stats
from mixem.distribution.distribution import Distribution
[docs]class NormalDistribution(Distribution):
"""Univariate normal distribution with parameters (mu, sigma)."""
def __init__(self, mu, sigma):
self.mu = mu
self.sigma = sigma
def log_density(self, data):
assert(len(data.shape) == 1), "Expect 1D data!"
return - (data - self.mu) ** 2 / (2 * self.sigma ** 2) - np.log(self.sigma) - 0.5 * np.log(2 * np.pi)
def estimate_parameters(self, data, weights):
assert(len(data.shape) == 1), "Expect 1D data!"
wsum = np.sum(weights)
self.mu = np.sum(weights * data) / wsum
self.sigma = np.sqrt(np.sum(weights * (data - self.mu) ** 2) / wsum)
def __repr__(self):
return "Norm[μ={mu:.4g}, σ={sigma:.4g}]".format(mu=self.mu, sigma=self.sigma)
[docs]class MultivariateNormalDistribution(Distribution):
"""Multivariate normal distribution with parameters (mu, Sigma)."""
def __init__(self, mu, sigma):
mu = np.array(mu)
sigma = np.array(sigma)
assert len(mu.shape) == 1, "Expect mu to be 1D vector!"
assert len(sigma.shape) == 2, "Expect sigma to be 2D matrix!"
assert sigma.shape[0] == sigma.shape[1], "Expect sigma to be a square matrix!"
self.mu = mu
self.sigma = sigma
def log_density(self, data):
return scipy.stats.multivariate_normal.logpdf(data, self.mu, self.sigma)
def estimate_parameters(self, data, weights):
self.mu = np.sum(data * weights[:, np.newaxis], axis=0) / np.sum(weights)
center_x = data - self.mu[np.newaxis, :]
# sigma = (np.diag(weights) @ center_x).T @ center_x / np.sum(weights)
self.sigma = np.dot(
np.dot(
np.diag(weights),
center_x
).T,
center_x
) / np.sum(weights)
def __repr__(self):
po = np.get_printoptions()
np.set_printoptions(precision=3)
try:
result = "MultiNorm[μ={mu}, σ={sigma}]".format(mu=self.mu, sigma=str(self.sigma).replace("\n", ","))
finally:
np.set_printoptions(**po)
return result