Source code for sklvq.models._gmlvq

from typing import Tuple
from typing import Union

import numpy as np
from sklearn.utils.validation import check_is_fitted, check_array

from . import LVQBaseClass
from ..objectives import GeneralizedLearningObjective

ModelParamsType = Tuple[np.ndarray, np.ndarray]

DISTANCES = [
    "adaptive-squared-euclidean",
]

SOLVERS = [
    "adaptive-moment-estimation",
    "broyden-fletcher-goldfarb-shanno",
    "limited-memory-bfgs",
    "steepest-gradient-descent",
    "waypoint-gradient-descent",
]


[docs]class GMLVQ(LVQBaseClass): r"""Generalized Matrix Learning Vector Quantization This model uses the :class:`sklvq.objectives.GeneralizedLearningObjective` as its objective function `[1]`_. In addition to learning the positions of the prototypes it learns a relevance matrix that is used in the distance functions `[2]`_. Parameters ---------- distance_type : {"adaptive-squared-euclidean"} or Class, default="squared-euclidean" Distance function that employs a relevance matrix in its calculation. - "adaptive-squared-euclidean" See :class:`sklvq.distances.AdaptiveSquaredEuclidean` distance_params : Dict, default=None Parameters passed to init of distance callable activation_type : {"identity", "sigmoid", "soft+", "swish"} or Class, default="sigmoid" Parameters passed to init of activation function. See the documentation of the activation functions for parameters and defaults. - "identity" See :class:`sklvq.activations.Identity` - "sigmoid" See :class:`sklvq.activations.Sigmoid` - "soft+" See :class:`sklvq.activations.SoftPlus` - "swish" See :class:`sklvq.activations.Swish` activation_params : Dict, default=None Parameters passed to init of activation function. See the documentation of activation functions for function dependent parameters and defaults. discriminant_type : {"relative-distance"} or Class, default = "relative-distance" The discriminant function. Note that different discriminant type may require to rewrite the ``decision_function`` and ``predict_proba`` methods. - "relative-distance" See :class:`sklvq.discriminants.RelativeDistance` discriminant_params : Dict, default=None Parameters passed to init of discriminant callable. See the documentation of the discriminant functions for parameters and defaults. solver_type : {"sgd", "wgd", "adam", "lbfgs", "bfgs"}, The solver used for optimization - "sgd" or "steepest-gradient-descent" See :class:`sklvq.solvers.SteepestGradientDescent`. - "wgd" or "waypoint-gradient-descent" See :class:`sklvq.solvers.WaypointGradientDescent`. - "adam" or "adaptive-moment-estimation" See :class:`sklvq.solvers.AdaptiveMomentEstimation`. - "bfgs" or "broyden-fletcher-goldfarb-shanno" Implementation from scipy package. - "lbfgs" or "limited-memory-bfgs" Implementation from scipy package. solver_params : dict, default=None Parameters passed to init of solvers. See the documentation of the solvers relevant parameters and defaults. prototype_init: "class-conditional-mean" or ndarray, default="class-conditional-mean" Default will initiate the prototypes to the class conditional mean with a small random offset. Custom numpy array can be passed to change the initial positions of the prototypes. prototype_n_per_class: int or np.ndarray, optional, default=1 Default will generate single prototype per class. In the case of unequal number of prototypes per class is needed, provide this as np.ndarray. For example, prototype_n_per_class = np.array([1, 6, 3]) this will result in one prototype for the first class, six for the second, and three for the third. Note that the order needs to be the same as the on in the classes\_ attribute, which is equal to calling np.unique(labels). relevance_init : {"identity", "random"} or np.ndarray, default="identity" Default will initiate the omega matrices to be the identity matrix. The rank of the matrix can be reduced by setting the ``relevance_n_components`` attribute `[3]`_. relevance_normalization: bool, optional, default=True Flag to indicate whether to normalize omega, whenever it is updated, such that the trace of the relevance matrix is equal to 1. relevance_n_components: str {"all"} or int, optional, default="all" For a square relevance matrix use the string "all" (default). For a rectangular relevance matrix use set the number of components explicitly by providing it as an int. random_state : int, RandomState instance, default=None Set the random number generation for reproducibility purposes. Used in random offset of prototypes and shuffling of the data in the solvers. Potentially, also used in the random generation of relevance matrix. force_all_finite : {True, "allow-nan"}, default=True Whether to raise an error on np.inf, np.nan, pd.NA in array. The possibilities are: - True: Force all values of array to be finite. - "allow-nan": accepts only np.nan and pd.NA values in array. Values cannot be infinite. Attributes ---------- classes_ : ndarray of shape (n_classes,) Class labels for each output. prototypes_ : ndarray of shape (n_protoypes, n_features) Positions of the prototypes after ``fit(X, labels)`` has been called. prototypes_labels_ : ndarray of shape (n_prototypes) Labels for each prototypes. Labels are indexes to ``classes_`` omega_: ndarray with size depending on initialization, default (n_features, n_features) Omega matrix that was found during training and defines the relevance matrix ``lambda_``. lambda_: ndarray of size (n_features, n_features) The relevance matrix ``omega_.T.dot(omega_)`` omega_hat_: ndarray The omega matrix found by the eigenvalue decomposition of the relevance matrix ``lambda_``. The eigenvectors (columns of ``omega_hat_``) can be used to transform the X `[3]`_. eigenvalues_: ndarray The corresponding eigenvalues to ``omega_hat_`` found by the eigenvalue decomposition of the relevance matrix ``lambda_`` References ---------- _`[1]` Sato, A., and Yamada, K. (1996) "Generalized Learning Vector Quantization." Advances in Neural Network Information Processing Systems, 423–429, 1996. _`[2]` Schneider, P., Biehl, M., & Hammer, B. (2009). "Adaptive Relevance Matrices in Learning Vector Quantization" Neural Computation, 21(12), 3532–3561, 2009. _`[3]` Bunte, K., Schneider, P., Hammer, B., Schleif, F.-M., Villmann, T., & Biehl, M. (2012). "Limited Rank Matrix Learning, discriminative dimension reduction and visualization." Neural Networks, 26, 159–173, 2012.""" classes_: np.ndarray prototypes_: np.ndarray prototypes_labels_: np.ndarray omega_: np.ndarray lambda_: np.ndarray omega_hat_: np.ndarray eigenvalues_: np.ndarray
[docs] def __init__( self, distance_type: Union[str, type] = "adaptive-squared-euclidean", distance_params: dict = None, activation_type: Union[str, type] = "sigmoid", activation_params: dict = None, discriminant_type: Union[str, type] = "relative-distance", discriminant_params: dict = None, solver_type: Union[str, type] = "steepest-gradient-descent", solver_params: dict = None, prototype_init: Union[str, np.ndarray] = "class-conditional-mean", prototype_n_per_class: Union[int, np.ndarray] = 1, relevance_init="identity", relevance_normalization: bool = True, relevance_n_components: Union[str, int] = "all", relevance_regularization: Union[int, float] = 0, random_state: Union[int, np.random.RandomState] = None, force_all_finite: Union[str, bool] = True, ): self.activation_type = activation_type self.activation_params = activation_params self.discriminant_type = discriminant_type self.discriminant_params = discriminant_params self.relevance_init = relevance_init self.relevance_normalization = relevance_normalization self.relevance_n_components = relevance_n_components self.relevance_regularization = relevance_regularization super(GMLVQ, self).__init__( distance_type, distance_params, DISTANCES, solver_type, solver_params, SOLVERS, prototype_init, prototype_n_per_class, random_state, force_all_finite, )
########################################################################################### # The "Getter" and "Setter" that are used by the solvers to set and get model params. ###########################################################################################
[docs] def set_variables(self, new_variables: np.ndarray) -> None: """ Modifies the ``self._variables`` by copying the values of ``new_variables`` into the memory of ``self._variables``. Parameters ---------- new_variables : ndarray 1d numpy array that contains all the model parameters in continuous memory """ np.copyto(self._variables, new_variables) if self.relevance_normalization: GMLVQ._normalise_omega(self.omega_)
[docs] def set_model_params(self, new_model_params: ModelParamsType): """ Changes the model's internal parameters. Copies the values of model_params into ``self.prototypes_`` and ``self.omega_`` therefor updating the ``self.variables_`` array. Also normalized the relevance matrix if necessary. Parameters ---------- new_model_params : tuple of ndarrays Shapes depend on initialization but in the case of a square relevance matrix: tuple((n_prototypes, n_features), (n_features, n_features)) """ new_prototypes, new_omega = new_model_params self.set_prototypes(new_prototypes) self.set_omega(new_omega) if self.relevance_normalization: GMLVQ._normalise_omega(self.omega_)
[docs] def get_model_params(self) -> ModelParamsType: """ Returns a tuple of all model parameters. In this case the prototypes and omega matrix. Returns ------- ndarray Returns a tuple of views, i.e., the prototypes and omega matrix. """ return self.prototypes_, self.omega_
########################################################################################### # Specific "getters" and "setters" for GMLVQ ###########################################################################################
[docs] def get_omega(self): """ Convenience function to return ``self.omega_`` Returns ------- ndarray, with shape depending on initialization of omega. """ return self.omega_
[docs] def set_omega(self, omega): """ Convenience function that makes sure to copy the value to ``self.omega_`` and not overwrite it. Parameters ---------- omega : ndarray with same shape as ``self.omega_`` """ np.copyto(self.omega_, omega)
########################################################################################### # Functions to transform the 1D variables array to model parameters and back ###########################################################################################
[docs] def to_model_params_view(self, var_buffer: np.ndarray) -> ModelParamsType: """ Parameters ---------- var_buffer : ndarray Array with the same size as the model's variables array as returned by ``get_variables()``. Returns ------- tuple Returns a tuple with the prototypes and omega matrix as ndarrays. """ return ( self.to_prototypes_view(var_buffer), self.to_omega(var_buffer), )
[docs] def to_prototypes_view(self, var_buffer: np.ndarray) -> np.ndarray: """ Returns a view (of the shape of the model's prototypes) into the provided variables buffer of the same size as the model's variables array. Parameters ---------- var_buffer : ndarray Array with the same size as the model's variables array as returned by ``get_variables()``. Returns ------- ndarray of shape (n_prototypes, n_features) Prototype view into the var_buffer. """ return var_buffer[: self._prototypes_size].reshape(self._prototypes_shape)
[docs] def to_omega(self, var_buffer: np.ndarray) -> np.ndarray: """ Returns a view (of the shape of the model's omega) into the provided variables buffer of the same size as the model's variables array. Parameters ---------- var_buffer : ndarray Array with the same size as the model's variables array as returned by ``get_variables()``. Returns ------- ndarray Shape depending on initialization but in case of a square matrix (n_features, n_features. """ return var_buffer[self._prototypes_size :].reshape(self._relevances_shape)
########################################################################################### # Solver Normalization functions ###########################################################################################
[docs] def normalize_variables(self, var_buffer: np.ndarray) -> None: """ Modifies the var_buffer as if it was the variables array provided by ``get_variables()``. Will select, reshape and normalize the correct parts of the variable buffer. Parameters ---------- var_buffer : ndarray Array with the same size as the model's variables array as returned by ``get_variables()``. """ (prototypes, omega) = self.to_model_params_view(var_buffer) self._normalize_prototypes(prototypes) self._normalise_omega(omega)
@staticmethod def _normalise_omega(omega: np.ndarray) -> None: np.divide(omega, np.sqrt(np.einsum("ji, ji", omega, omega)), out=omega) ########################################################################################### # Solver helper functions ###########################################################################################
[docs] def add_partial_gradient(self, gradient, partial_gradient, i_prototype) -> None: """ Adds the partial gradient to the correct part of the gradient, which depends on ``i_prototype``. Parameters ---------- gradient : ndarray Same shape as the ``get_variables()`` would return. partial_gradient : ndarray 1d array containing the partial gradient. i_prototype : int The index of the prototype to which the partial gradient was computed. """ n_features = self.n_features_in_ prots_view = self.to_prototypes_view(gradient) np.add( prots_view[i_prototype, :], partial_gradient[:n_features], out=prots_view[i_prototype, :], ) omega_view = gradient[self.prototypes_.size :] np.add(omega_view, partial_gradient[n_features:], out=omega_view)
[docs] def mul_step_size( self, step_sizes: Union[int, float, np.ndarray], gradient: np.ndarray ) -> None: """ If step sizes is a scalar value just multiplies the gradient with the step size. If it is an array (with same length as number of model parameters) each model parameter is multiplied by its own step size. Parameters ---------- step_sizes : float or ndarray The scalar or list of values containing the step sizes. gradient : ndarray Same shape as the ``get_variables()`` would return. """ if isinstance(step_sizes, int) | isinstance(step_sizes, float): gradient *= step_sizes return # else it's a np.ndarray... if isinstance(step_sizes, np.ndarray): if step_sizes.size > 1: prototypes, omega = self.to_model_params_view(gradient) prototypes *= step_sizes[0] omega *= step_sizes[1]
# if size is more than 2 it just ignores the additional values. ########################################################################################### # Initialization functions ########################################################################################### def _init_variables(self) -> None: self._variables = np.empty( self._prototypes_size + self._relevances_size, dtype="float64", order="C", ) def _check_model_params(self): self._check_prototype_params() self._check_relevances_params() def _init_model_params(self, X, y) -> None: self._init_prototypes(X, y) self._init_relevances() def _check_relevances_params(self): relevance_normalization = self.relevance_normalization if not isinstance(relevance_normalization, bool): raise ValueError("Provided normalization is invalid.") relevance_n_components = self.relevance_n_components if isinstance(relevance_n_components, str): if relevance_n_components == "all": self._relevances_shape = (self.n_features_in_, self.n_features_in_) else: raise ValueError("Provided n_components is invalid.") elif isinstance(relevance_n_components, int): if ( self.relevance_n_components >= 1 and relevance_n_components <= self.n_features_in_ ): self._relevances_shape = (relevance_n_components, self.n_features_in_) else: raise ValueError("Provided n_components is invalid.") else: raise ValueError("Provided n_components is invalid.") self._relevances_size = np.prod(self._relevances_shape) def _init_relevances(self): self.omega_ = self.to_omega(self._variables) if isinstance(self.relevance_init, str): if self.relevance_init == "identity": self.set_omega(np.eye(*self._relevances_shape)) elif self.relevance_init == "random": self.set_omega( self.random_state_.uniform( low=0, high=1, size=self._relevances_shape ) ) else: raise ValueError("Provided relevance_init is invalid.") else: raise ValueError("Provided relevance_init is invalid.") if self.relevance_normalization: GMLVQ._normalise_omega(self.omega_) def _init_objective(self): self._objective = GeneralizedLearningObjective( self.activation_type, self.activation_params, self.discriminant_type, self.discriminant_params, ) ########################################################################################### # Other Algorithm specific stuff. ########################################################################################### def _after_fit(self, X: np.ndarray, y: np.ndarray): self.lambda_ = GMLVQ._compute_lambda(self.omega_) # Eigenvalues and column eigenvectors return in ascending order eigenvalues, omega_hat = np.linalg.eigh(self.lambda_) # Flip (reverse the order to descending) before assigning. self.eigenvalues_ = np.flip(eigenvalues) # eigenvectors are column matrix in ascending order. Flip the columns and transpose the matrix # to get the descending ordered row matrix. self.omega_hat_ = np.flip(omega_hat, axis=1).T @staticmethod def _compute_lambda(omega): # Equivalent to omega.T.dot(omega), but faster (?) return np.einsum("ji, jk -> ik", omega, omega) ########################################################################################### # Transformer related functions ###########################################################################################
[docs] def fit_transform( self, data: np.ndarray, y: np.ndarray, **transform_params ) -> np.ndarray: r""" Parameters ---------- data : ndarray with shape (n_samples, n_features) Data used for fit and that will be transformed. y : np.ndarray with length (n_samples) Labels corresponding to the X samples. transform_params : Parameters passed to transform function Returns ------- The data projected on columns of ``omega_hat_`` with shape (n_samples, n_columns) """ return self.fit(data, y).transform(data, **transform_params)
[docs] def transform(self, X: np.ndarray, scale: bool = False) -> np.ndarray: r""" Parameters ---------- X : np.ndarray with shape (n_samples, n_features) Data that needs to be transformed scale : {True, False}, default = False Controls if the eigenvectors the data is projected on are scaled by the square root of their eigenvalues. Returns ------- The data projected on columns of ``omega_hat_`` with shape (n_samples, n_columns) """ X = check_array(X) check_is_fitted(self) transformation_matrix = self.omega_hat_ if scale: transformation_matrix = ( np.sqrt(np.absolute(self.eigenvalues_[:, None])) * transformation_matrix ) data_new = X.dot(transformation_matrix.T) return data_new