Source code for pysensmcda.probabilistic.perturbed_matrix

# Copyright (C) 2024 Jakub Więckowski

import numpy as np
from ..validator import Validator
from ..utils import memory_guard

[docs] @memory_guard def perturbed_matrix(matrix: np.ndarray, simulations: int, precision: int = 6, perturbation_scale: float | np.ndarray = 0.1) -> np.ndarray: """ Generate perturbed decision matrices based on the given initial decision matrix using random perturbation based on uniform distribution. Parameters: ------------ matrix : ndarray 2D array representing the initial decision matrix. simulations : int Number of perturbed decision matrix simulations to generate. precision : int, optional, default=6 Precision for rounding the perturbed values from the decision matrix. perturbation_scale : float | np.ndarray, optional, default=0.1 Scale for random perturbation added to each value from the decision matrix. If float, then all decision matrix is modeled with the same perturbation scale. If ndarray, then each criterion is modeled with a given perturbation scale. Returns: --------- ndarray A ndarray of simulations length with perturbed decision matrices based on the given initial decision matrix. Examples: ---------- Example 1: Run with default parameters >>> matrix = np.array([ [4, 3, 7], [1, 9, 6], [7, 5, 3] ]) >>> simulations = 1000 >>> results = perturbed_matrix(matrix, simulations) >>> for r in results: ... print(r) Example 2: Run with given precision and perturbation scale >>> matrix = np.array([ [4, 3, 7], [1, 9, 6], [7, 5, 3] ]) >>> simulations = 500 >>> precision = 3 >>> perturbation_scale = 1 >>> results = perturbed_matrix(matrix, simulations, precision, perturbation_scale) >>> for r in results: ... print(r) Example 3: Run with perturbation scale defined for each column >>> matrix = np.array([ [4, 3, 7], [1, 9, 6], [7, 5, 3] ]) >>> simulations = 100 >>> precision = 3 >>> perturbation_scale = np.array([0.5, 1, 0.4]) >>> results = perturbed_matrix(matrix, simulations, precision, perturbation_scale) >>> for r in results: ... print(r) Example 4: Run with 2D perturbation scale array >>> matrix = np.array([ [4, 3, 7], [1, 9, 6], [7, 5, 3] ]) >>> simulations = 100 >>> precision = 3 >>> perturbation_scale = np.array([ [0.4, 0.5, 1], [0.7, 0.3, 1.2], [0.5, 0.1, 1.5] ]) >>> results = perturbed_matrix(matrix, simulations, precision, perturbation_scale) >>> for r in results: ... print(r) """ Validator.is_type_valid(matrix, np.ndarray, 'matrix') Validator.is_dimension_valid(matrix, 2, 'matrix') Validator.is_type_valid(simulations, (int, np.integer), 'simulations') Validator.is_positive_value(simulations, var_name='simulations') Validator.is_type_valid(precision, (int, np.integer), 'precision') Validator.is_positive_value(precision, var_name='precision') Validator.is_type_valid(perturbation_scale, (int, np.integer, float, np.floating, np.ndarray), 'perturbation_scale') if isinstance(perturbation_scale, (float, np.floating, int, np.integer)): perturbation_scale = np.full(matrix.shape[0], perturbation_scale) elif isinstance(perturbation_scale, np.ndarray): if perturbation_scale.ndim == 1: Validator.is_shape_equal(matrix.shape[1], perturbation_scale.shape[0], custom_message="Number of columns in 'matrix' and length of 'perturbation_scale' are different") elif perturbation_scale.ndim == 2: Validator.is_shape_equal(matrix.shape, perturbation_scale.shape, custom_message="Shapes of 'matrix' and 'perturbation_scale' are different") modified_matrices = [] for _ in range(simulations): perturbation = np.random.uniform(-perturbation_scale, perturbation_scale, (matrix.shape[0], matrix.shape[1])) modified_matrix = matrix + perturbation modified_matrices.append(list(np.round(modified_matrix, precision))) return np.array(modified_matrices)