Source code for statsmodels.multivariate.manova
"""Multivariate analysis of variance
author: Yichuan Liu
"""
from statsmodels.compat.pandas import Substitution
import numpy as np
from statsmodels.base.model import Model
from statsmodels.formula._manager import FormulaManager
from .multivariate_ols import (
MultivariateTestResults,
_hypotheses_doc,
_multivariate_ols_fit,
_multivariate_ols_test,
)
__docformat__ = 'restructuredtext en'
[docs]
class MANOVA(Model):
"""
Multivariate Analysis of Variance
The implementation of MANOVA is based on multivariate regression and does
not assume that the explanatory variables are categorical. Any type of
variables as in regression is allowed.
Parameters
----------
endog : array_like
Dependent variables. A nobs x k_endog array where nobs is
the number of observations and k_endog is the number of dependent
variables.
exog : array_like
Independent variables. A nobs x k_exog array where nobs is the
number of observations and k_exog is the number of independent
variables. An intercept is not included by default and should be added
by the user. Models specified using a formula include an intercept by
default.
Attributes
----------
endog : ndarray
See Parameters.
exog : ndarray
See Parameters.
Notes
-----
MANOVA is used though the `mv_test` function, and `fit` is not used.
The ``from_formula`` interface is the recommended method to specify
a model and simplifies testing without needing to manually configure
the contrast matrices.
References
----------
.. [*] ftp://public.dhe.ibm.com/software/analytics/spss/documentation/
statistics/20.0/en/client/Manuals/IBM_SPSS_Statistics_Algorithms.pdf
"""
_formula_max_endog = None
def __init__(self, endog, exog, missing="none", hasconst=None, **kwargs):
if len(endog.shape) == 1 or endog.shape[1] == 1:
raise ValueError(
"There must be more than one dependent variable" " to fit MANOVA!"
)
super().__init__(endog, exog, missing=missing, hasconst=hasconst, **kwargs)
self._fittedmod = _multivariate_ols_fit(self.endog, self.exog)
[docs]
def fit(self):
raise NotImplementedError('fit is not needed to use MANOVA. Call'
'mv_test directly on a MANOVA instance.')
[docs]
@Substitution(hypotheses_doc=_hypotheses_doc)
def mv_test(self, hypotheses=None, skip_intercept_test=False):
"""
Linear hypotheses testing
Parameters
----------
%(hypotheses_doc)s
skip_intercept_test : bool
If true, then testing the intercept is skipped, the model is not
changed.
Note: If a term has a numerically insignificant effect, then
an exception because of emtpy arrays may be raised. This can
happen for the intercept if the data has been demeaned.
Returns
-------
results: MultivariateTestResults
Notes
-----
Testing the linear hypotheses
L * params * M = 0
where `params` is the regression coefficient matrix for the
linear model y = x * params
If the model is not specified using the formula interfact, then the
hypotheses test each included exogenous variable, one at a time. In
most applications with categorical variables, the ``from_formula``
interface should be preferred when specifying a model since it
provides knowledge about the model when specifying the hypotheses.
"""
if hypotheses is None:
if (
hasattr(self, "data")
and self.data is not None
and hasattr(self.data, "model_spec")
):
# TODO: patsy migration
mgr = FormulaManager()
terms = mgr.get_term_name_slices(self.data.model_spec)
hypotheses = []
for key in terms:
if skip_intercept_test and (
key == "Intercept" or key == mgr.intercept_term
):
continue
L_contrast = np.eye(self.exog.shape[1])[terms[key], :]
test_name = str(key)
if key == mgr.intercept_term:
test_name = "Intercept"
hypotheses.append([test_name, L_contrast, None])
else:
hypotheses = []
for i in range(self.exog.shape[1]):
name = "x%d" % (i)
L = np.zeros([1, self.exog.shape[1]])
L[0, i] = 1
hypotheses.append([name, L, None])
results = _multivariate_ols_test(
hypotheses, self._fittedmod, self.exog_names, self.endog_names
)
return MultivariateTestResults(results, self.endog_names, self.exog_names)
Last update:
Jan 20, 2025