From 9dc21384cb570f09bee92b9b01de898a2c36c097 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marc=20Boull=C3=A9?= Date: Wed, 11 Oct 2023 17:46:48 +0200 Subject: [PATCH] Refactor remove class CCCoclusteringOptimizer CCCoclusteringOptimizer - supression de la classe (pour l'instant commentee) - remplacee en redesignant l'interaction entre KWAttributeStats et KWDataGridOptimizer pour la gestion des HandleOptimizationStep - class KWAttributeSubsetStats : public KWDataPreparationStats - methode virtuelle HandleOptimizationStep - KWDataGridOptimizer - Set|GetAttributeSubsetStats: pour parametrer le "builder" qui demande l'optimisation, et lui rediriger les HandleOptimizationStep - CCCoclusteringBuilder utilise desormais directement un KWDataGridOptimizer NEWKWDataGridOptimizer - duplication de KWDataGridOptimizer temporaire, pour tester un refactoring majeur - appele dans KWDataGridOptimizer::OptimizeDataGrid selon un booleen bNewProto - tests elementaires --- .../KWAttributeSubsetStats.cpp | 6 + .../KWAttributeSubsetStats.h | 5 + .../KWDataPreparation/KWDataGridManager.cpp | 2 +- .../KWDataPreparation/KWDataGridOptimizer.cpp | 37 + .../KWDataPreparation/KWDataGridOptimizer.h | 17 +- .../NEWKWDataGridOptimizer.cpp | 2624 +++++++++++++++++ .../NEWKWDataGridOptimizer.h | 336 +++ src/Learning/MODL/MODL.cpp | 1 + .../CCCoclusteringBuilder.cpp | 46 +- .../MODL_Coclustering/CCCoclusteringBuilder.h | 9 +- .../MODL_Coclustering/MODL_Coclustering.cpp | 2 +- 11 files changed, 3032 insertions(+), 53 deletions(-) create mode 100644 src/Learning/KWDataPreparation/NEWKWDataGridOptimizer.cpp create mode 100644 src/Learning/KWDataPreparation/NEWKWDataGridOptimizer.h diff --git a/src/Learning/KWDataPreparation/KWAttributeSubsetStats.cpp b/src/Learning/KWDataPreparation/KWAttributeSubsetStats.cpp index 814650624..43a517330 100644 --- a/src/Learning/KWDataPreparation/KWAttributeSubsetStats.cpp +++ b/src/Learning/KWDataPreparation/KWAttributeSubsetStats.cpp @@ -611,6 +611,12 @@ KWDataGrid* KWAttributeSubsetStats::CreateDataGrid(const KWTupleTable* tupleTabl return dataGrid; } +void KWAttributeSubsetStats::HandleOptimizationStep(const KWDataGrid* optimizedDataGrid, + const KWDataGrid* initialGranularizedDataGrid, + boolean bIsLastSaving) const +{ +} + boolean KWAttributeSubsetStats::GetPregranularizedNumericalAttributes() { return bPregranularizedNumericalAttributes; diff --git a/src/Learning/KWDataPreparation/KWAttributeSubsetStats.h b/src/Learning/KWDataPreparation/KWAttributeSubsetStats.h index 25440ba3c..6ceb1e3d5 100644 --- a/src/Learning/KWDataPreparation/KWAttributeSubsetStats.h +++ b/src/Learning/KWDataPreparation/KWAttributeSubsetStats.h @@ -91,6 +91,11 @@ class KWAttributeSubsetStats : public KWDataPreparationStats // Memoire: appartient a l'appelant KWDataGrid* CreateDataGrid(const KWTupleTable* tupleTable); + // Methode appelee lors de l'optimisation a chaque etape d'optimisation + // Implementation vide par defaut + virtual void HandleOptimizationStep(const KWDataGrid* optimizedDataGrid, + const KWDataGrid* initialGranularizedDataGrid, boolean bIsLastSaving) const; + // Parametrage avance // Pre-granularisation des attributs numeriques cible (regression) et des attributs numeriques explicatifs en // analyse non supervisee (co-clustering) Cette pre-granularisation permet : diff --git a/src/Learning/KWDataPreparation/KWDataGridManager.cpp b/src/Learning/KWDataPreparation/KWDataGridManager.cpp index d479a231e..3c70c2a8d 100644 --- a/src/Learning/KWDataPreparation/KWDataGridManager.cpp +++ b/src/Learning/KWDataPreparation/KWDataGridManager.cpp @@ -1900,7 +1900,7 @@ void KWDataGridManager::BuildUnivariateDataGridFromGranularizedPartition(KWDataG KWDGAttribute* targetAttribute; KWDGAttribute* sourceAttribute; - require(0 < nAttributeIndex and nAttributeIndex < sourceDataGrid->GetAttributeNumber()); + require(0 <= nAttributeIndex and nAttributeIndex < sourceDataGrid->GetAttributeNumber()); // Initialisation de la grille cible a une variable InitialiseDataGrid(sourceDataGrid, univariateTargetDataGrid, 1); diff --git a/src/Learning/KWDataPreparation/KWDataGridOptimizer.cpp b/src/Learning/KWDataPreparation/KWDataGridOptimizer.cpp index b5ff2f66c..1d3a7709e 100644 --- a/src/Learning/KWDataPreparation/KWDataGridOptimizer.cpp +++ b/src/Learning/KWDataPreparation/KWDataGridOptimizer.cpp @@ -16,6 +16,7 @@ KWDataGridOptimizer::KWDataGridOptimizer() // CH IV Begin initialVarPartDataGrid = NULL; // CH IV end + attributeSubsetStatsHandler = NULL; } KWDataGridOptimizer::~KWDataGridOptimizer() {} @@ -29,6 +30,7 @@ void KWDataGridOptimizer::Reset() // CH IV Begin initialVarPartDataGrid = NULL; // CH IV end + attributeSubsetStatsHandler = NULL; } void KWDataGridOptimizer::SetDataGridCosts(const KWDataGridCosts* kwdgcCosts) @@ -103,6 +105,27 @@ double KWDataGridOptimizer::OptimizeDataGrid(const KWDataGrid* initialDataGrid, int nLastExploredGranularity; ALString sTmp; + // CH IV Refactoring : DDDDD + // Test du remplacement de la methode actuelle, par son proto + boolean bNewPROTO = false; + if (bNewPROTO) + { + NEWKWDataGridOptimizer newDataGridOptimizer; + + // Recopie du parametrage courant + newDataGridOptimizer.Reset(); + newDataGridOptimizer.GetParameters()->CopyFrom(&optimizationParameters); + newDataGridOptimizer.SetClassStats(GetClassStats()); + newDataGridOptimizer.SetDataGridCosts(GetDataGridCosts()); + newDataGridOptimizer.bCleanNonInformativeVariables = bCleanNonInformativeVariables; + newDataGridOptimizer.initialVarPartDataGrid = initialVarPartDataGrid; + newDataGridOptimizer.SetAttributeSubsetStats(attributeSubsetStatsHandler); + + // Optimisation + dBestCost = newDataGridOptimizer.OptimizeDataGrid(initialDataGrid, optimizedDataGrid); + return dBestCost; + } + dGranularityBestCost = DBL_MAX; dBestMergedCost = dGranularityBestCost; dTotalTime = 0; @@ -689,6 +712,20 @@ void KWDataGridOptimizer::HandleOptimizationStep(const KWDataGrid* optimizedData const KWDataGrid* initialGranularizedDataGrid, boolean bIsLastSaving) const { + // Integration de la granularite + if (attributeSubsetStatsHandler != NULL) + attributeSubsetStatsHandler->HandleOptimizationStep(optimizedDataGrid, initialGranularizedDataGrid, + bIsLastSaving); +} + +void KWDataGridOptimizer::SetAttributeSubsetStats(const KWAttributeSubsetStats* attributeSubsetStats) +{ + attributeSubsetStatsHandler = attributeSubsetStats; +} + +const KWAttributeSubsetStats* KWDataGridOptimizer::GetAttributeSubsetStats() +{ + return attributeSubsetStatsHandler; } void KWDataGridOptimizer::PostOptimizeGranularity(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid, diff --git a/src/Learning/KWDataPreparation/KWDataGridOptimizer.h b/src/Learning/KWDataPreparation/KWDataGridOptimizer.h index e554015f3..f4ca45baf 100644 --- a/src/Learning/KWDataPreparation/KWDataGridOptimizer.h +++ b/src/Learning/KWDataPreparation/KWDataGridOptimizer.h @@ -18,6 +18,7 @@ class CCCoclusteringOptimizer; #include "SortedList.h" #include "Profiler.h" #include "Timer.h" +#include "NEWKWDataGridOptimizer.h" ////////////////////////////////////////////////////////////////////////////////// // Classe KWDataGridOptimizer @@ -83,16 +84,15 @@ class KWDataGridOptimizer : public Object // Methode appelee lors de l'optimisation a la fin de chaque etape d'optimisation // Permet par exemple de passer en mode anytime, en memorisant chaque solution intermediaire - // (solution resultant de random + pre-optmisation + optimisation + post-optimisation, donc optimum local) - // Par defaut: ne fait rien - // Transmission de la grille initiale granularisee - // initialDataGrid est la grille initiale granularise en mode granularisation sinon NULL ce qui sous-entend - // qu'il n'y a pas de granularisation bIsLastSaving : derniere memorisation apres le traitement de la - // granularite maximale qui justifie la memorisation de la grille meme si le cout n'a pas ete ameliore pour - // recalculer les informations de la hierachie + // Par defaut: redirige vers la methode HandleOptimizationStep attributeSubsetStats s'il est parametre virtual void HandleOptimizationStep(const KWDataGrid* optimizedDataGrid, const KWDataGrid* initialGranularizedDataGrid, boolean bIsLastSaving) const; + // Parametrage du contexte de gestion de la partie anytime de l'optimisation + // Permet de rediriger la methode HandleOptimizationStep vers celle du attributeSubsetStats + void SetAttributeSubsetStats(const KWAttributeSubsetStats* attributeSubsetStats); + const KWAttributeSubsetStats* GetAttributeSubsetStats(); + ////////////////////////////////////////////////////////////////// // Gestion d'un profiler dedie a l'optimisation des grilles // Ce profiler doit etre demarre depuis le point d'entree de l'optimisation, @@ -168,6 +168,9 @@ class KWDataGridOptimizer : public Object // Epsilon d'optimisation double dEpsilon; + // Contexte de gestion de la partie anytime de l'optimisation + const KWAttributeSubsetStats* attributeSubsetStatsHandler; + // Profiler static Profiler profiler; }; diff --git a/src/Learning/KWDataPreparation/NEWKWDataGridOptimizer.cpp b/src/Learning/KWDataPreparation/NEWKWDataGridOptimizer.cpp new file mode 100644 index 000000000..874a1f806 --- /dev/null +++ b/src/Learning/KWDataPreparation/NEWKWDataGridOptimizer.cpp @@ -0,0 +1,2624 @@ +// Copyright (c) 2023 Orange. All rights reserved. +// This software is distributed under the BSD 3-Clause-clear License, the text of which is available +// at https://spdx.org/licenses/BSD-3-Clause-Clear.html or see the "LICENSE" file for more details. + +#include "NEWKWDataGridOptimizer.h" + +////////////////////////////////////////////////////////////////////////////////// +// Classe NEWKWDataGridOptimizer + +NEWKWDataGridOptimizer::NEWKWDataGridOptimizer() +{ + classStats = NULL; + dataGridCosts = NULL; + bCleanNonInformativeVariables = false; + dEpsilon = 1e-6; + // CH IV Begin + initialVarPartDataGrid = NULL; + // CH IV end + attributeSubsetStatsHandler = NULL; +} + +NEWKWDataGridOptimizer::~NEWKWDataGridOptimizer() {} + +void NEWKWDataGridOptimizer::Reset() +{ + classStats = NULL; + dataGridCosts = NULL; + bCleanNonInformativeVariables = false; + dEpsilon = 1e-6; + // CH IV Begin + initialVarPartDataGrid = NULL; + // CH IV end + attributeSubsetStatsHandler = NULL; +} + +void NEWKWDataGridOptimizer::SetDataGridCosts(const KWDataGridCosts* kwdgcCosts) +{ + dataGridCosts = kwdgcCosts; +} + +const KWDataGridCosts* NEWKWDataGridOptimizer::GetDataGridCosts() const +{ + return dataGridCosts; +} + +KWDataGridOptimizerParameters* NEWKWDataGridOptimizer::GetParameters() +{ + return &optimizationParameters; +} + +void NEWKWDataGridOptimizer::SetClassStats(KWClassStats* stats) +{ + classStats = stats; +} + +KWClassStats* NEWKWDataGridOptimizer::GetClassStats() const +{ + return classStats; +} +// CH IV Begin +void NEWKWDataGridOptimizer::SetInitialVarPartDataGrid(KWDataGrid* refDataGrid) +{ + require(refDataGrid != NULL); + require(refDataGrid->IsVarPartDataGrid()); + initialVarPartDataGrid = refDataGrid; +} + +KWDataGrid* NEWKWDataGridOptimizer::GetInitialVarPartDataGrid() const +{ + return initialVarPartDataGrid; +} +// CH IV End +double NEWKWDataGridOptimizer::OptimizeDataGrid(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid) const + +{ + boolean bDisplayResults = false; + boolean bDisplayGranularities = false; + KWDataGrid granularizedDataGrid; + KWDataGrid* granularizedOptimizedDataGrid; + KWDataGridManager dataGridManager; + double dGranularityBestCost; + double dBestCost; + double dTotalTime; + int nGranularityIndex; + int nGranularityMax; + int nValueNumber; + boolean bIsLastGranularity; + boolean bOptimizationNeeded; + boolean bWithMemoryMode; + ObjectDictionary odQuantileBuilders; + IntVector ivPreviousPartNumber; + IntVector ivCurrentPartNumber; + int nAttribute; + boolean bIsGranularitySelected; + double dRequiredIncreasingCoefficient; + IntVector ivMaxPartNumbers; + int nMinValueNumber = 500; + boolean bIsGranularityMaxThresholded; + double dBestMergedCost; + double dMergedCost; + KWDataGrid granularizedPostMergedOptimizedDataGrid; + KWDataGrid partitionedReferenceGranularizedPostMergedDataGrid; + double dFusionDeltaCost; + int nCurrentExploredGranularity; + int nLastExploredGranularity; + ALString sTmp; + + dGranularityBestCost = DBL_MAX; + dBestMergedCost = dGranularityBestCost; + dTotalTime = 0; + nValueNumber = initialDataGrid->GetGridFrequency(); + bWithMemoryMode = false; + + // Controle de la graine aleatoire pour avoir des resultats reproductibles + SetRandomSeed(1); + + // Construction d'une grille terminale pour la solution initiale + dBestCost = InitializeWithTerminalDataGrid(initialDataGrid, optimizedDataGrid); + + if (bDisplayResults) + cout << "KWOptimize : Cout grille terminale independant de la granularite " << dBestCost << endl; + + if (bDisplayResults) + { + cout << "KWOptimize :Grille initiale avant optimisation" << endl; + initialDataGrid->Write(cout); + } + + // On determine si on peut potentiellement faire mieux que la grille terminale + bOptimizationNeeded = true; + if ((initialDataGrid->GetTargetAttribute() == NULL and initialDataGrid->GetTargetValueNumber() == 1) or + (initialDataGrid->GetTargetAttribute() == NULL and initialDataGrid->GetInformativeAttributeNumber() == 0) or + (initialDataGrid->GetTargetAttribute() != NULL and + initialDataGrid->GetTargetAttribute()->GetPartNumber() <= 1) or + (initialDataGrid->GetTargetAttribute() != NULL and initialDataGrid->GetInformativeAttributeNumber() <= 1)) + bOptimizationNeeded = false; + + // Cas ou la grille terminale est ameliorable + if (bOptimizationNeeded) + { + // On parcourt les differentes granularites + + // Initialisation de la granularite maximale avec seuillage eventuel de la granularite max pour reduire + // la complexite algorithmique + bIsGranularityMaxThresholded = false; + // Cas d'une une grille 2D supervise avec un attribut a predire avec groupage de la cible + // (classification avec groupage ou regression) et un attribut explicatif numerique dont le nombre de + // valeurs potentielle (= nombre d'instances) est > nMinValueNumber + if (initialDataGrid->GetTargetAttribute() != NULL and initialDataGrid->GetTargetValueNumber() == 0 and + initialDataGrid->GetAttributeNumber() == 2 and + initialDataGrid->GetAttributeAt(0)->GetAttributeType() == KWType::Continuous and + (nValueNumber > nMinValueNumber)) + bIsGranularityMaxThresholded = true; + // Cas d'une classification supervisee simple avec une paire de variables (numerique ou categorielle) + // pour lesquelles le nombre de valeurs potentielles (= nombre d'instances) est > nMinValueNumber + else if (initialDataGrid->GetTargetValueNumber() > 0 and initialDataGrid->GetAttributeNumber() == 2 and + nValueNumber > nMinValueNumber) + bIsGranularityMaxThresholded = true; + + // Cas sans seuillage + if (not bIsGranularityMaxThresholded) + nGranularityMax = (int)ceil(log(nValueNumber * 1.0) / log(2.0)); + // Sinon seuillage + else + nGranularityMax = + (int)ceil(log(nMinValueNumber + sqrt((nValueNumber - nMinValueNumber) * + log(nValueNumber - nMinValueNumber) / log(2.0))) / + log(2.0)); + + // Initialisation + nGranularityIndex = 1; + bIsLastGranularity = false; + + // Initialisation du facteur d'accroissement requis entre deux partitions traitees + dRequiredIncreasingCoefficient = 2; + + // Initialisation des quantiles builders a partir de la grille source + dataGridManager.SetSourceDataGrid(initialDataGrid); + dataGridManager.InitializeQuantileBuilders(&odQuantileBuilders, &ivMaxPartNumbers); + + if (bDisplayResults) + cout << "ivMaxPartNumbers Granularisation\t" << ivMaxPartNumbers << flush; + + // Initialisation des vecteurs de nombre de parties courant et precedent + ivPreviousPartNumber.SetSize(ivMaxPartNumbers.GetSize()); + ivCurrentPartNumber.SetSize(ivMaxPartNumbers.GetSize()); + + // Initialisation + nCurrentExploredGranularity = -1; + nLastExploredGranularity = -1; + + // Parcours des granularites + while (nGranularityIndex <= nGranularityMax and not bIsLastGranularity) + { + // Arret si interruption utilisateur + if (TaskProgression::IsInterruptionRequested()) + break; + + // Granularisation de la grille initiale + dataGridManager.SetSourceDataGrid(initialDataGrid); + dataGridManager.ExportGranularizedDataGrid(&granularizedDataGrid, nGranularityIndex, + &odQuantileBuilders); + + // Etude du nombre de parties attribut par attribut pour decider du lancement de l'optimisation + // ou non pour cette granularite Parcours des attributs + for (nAttribute = 0; nAttribute < granularizedDataGrid.GetAttributeNumber(); nAttribute++) + { + // Memorisation du nombre de parties de l'attribut granularise + ivCurrentPartNumber.SetAt( + nAttribute, cast(KWDGAttribute*, granularizedDataGrid.GetAttributeAt(nAttribute)) + ->GetPartNumber()); + } + bIsLastGranularity = true; + + // Si on n'a pas encore atteint la granularite max + if (nGranularityIndex < nGranularityMax) + { + for (nAttribute = 0; nAttribute < granularizedDataGrid.GetAttributeNumber(); + nAttribute++) + { + // Cas ou le nombre de parties de l'attribut courant est inferieur au nombre max + // de parties de l'attribut + if (ivCurrentPartNumber.GetAt(nAttribute) < ivMaxPartNumbers.GetAt(nAttribute)) + { + bIsLastGranularity = false; + break; + } + } + } + + // Cas ou cette granularite sera la derniere traitee + if (bIsLastGranularity) + // On positionne l'index de granularite au maximum afin que l'affichage soit adapte a ce + // cas + granularizedDataGrid.SetGranularity(nGranularityMax); + + bIsGranularitySelected = false; + for (nAttribute = 0; nAttribute < granularizedDataGrid.GetAttributeNumber(); nAttribute++) + { + // Cas d'accroissement suffisant du nombre de parties + if ((ivCurrentPartNumber.GetAt(nAttribute) >= + ivPreviousPartNumber.GetAt(nAttribute) * dRequiredIncreasingCoefficient) and + (ivCurrentPartNumber.GetAt(nAttribute) * dRequiredIncreasingCoefficient <= + ivMaxPartNumbers.GetAt(nAttribute))) + { + bIsGranularitySelected = true; + break; + } + } + + // On ne traite pas les grilles avec un seul attribut informatif + // Dans le cas d'une grille avec un attribut instances et un attribut VarPart, tant que + // l'attribut instances ne contient qu'une seule partie, la granularite n'est pas selectionnee. + // La granularisation de l'attribut instances est reduite au fourre-tout (1 seule partie) + // tant que le nombre d'observations par instances n'est pas superieur a l'effectif minimal N/2^G + // Pour un nombre d'observations egal au nombre de variables pour toutes les instances, + // il faut atteindre G tel que G > Gmax - log(K) / log(2) + if (granularizedDataGrid.GetInformativeAttributeNumber() <= 1) + bIsGranularitySelected = false; + + // Cas du traitement de la granularite courante + if (bIsGranularitySelected or bIsLastGranularity) + { + // Memorisation des granularites exploitees + nLastExploredGranularity = nCurrentExploredGranularity; + nCurrentExploredGranularity = nGranularityIndex; + + // Initialisation de la grille granularisee optimisee a la grille terminale + granularizedOptimizedDataGrid = new KWDataGrid; + dGranularityBestCost = InitializeWithTerminalDataGrid(&granularizedDataGrid, + granularizedOptimizedDataGrid); + + if (bDisplayResults) + { + cout << "KWOptimize :Cout Grille initiale granularisee pour granularite = " + << IntToString(nGranularityIndex) << "\t" + << dataGridCosts->ComputeDataGridTotalCost(&granularizedDataGrid) << endl; + granularizedDataGrid.Write(cout); + granularizedDataGrid.WriteAttributes(cout); + granularizedDataGrid.WriteAttributeParts(cout); + cout << "KWOptimize : Cout grille terminale pour cette granularite\t" + << dGranularityBestCost << endl; + } + + // Cas avec memoire (pas par defaut) : on part a la granularite courante de la meilleure + // grille rencontree aux granularites precedentes, si elle est meilleure que la grille + // initiale a cette granularite + if (bWithMemoryMode and dBestCost < dGranularityBestCost) + dataGridManager.CopyDataGrid(optimizedDataGrid, granularizedOptimizedDataGrid); + + // Parametrage du profiling + if (bIsLastGranularity) + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Optimize last granularity"); + else + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Optimize granularity"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Granularity index", + IntToString(nGranularityIndex)); + if (optimizedDataGrid->IsVarPartDataGrid()) + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString( + "VarPart granularity", + IntToString( + optimizedDataGrid->GetInnerAttributes()->GetVarPartGranularity())); + + // Optimisation de la grille granularisee + dGranularityBestCost = + OptimizeGranularizedDataGrid(&granularizedDataGrid, granularizedOptimizedDataGrid, + bIsLastGranularity, dTotalTime); + if (bIsLastGranularity) + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Optimize last granularity"); + else + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Optimize granularity"); + + if (bDisplayResults) + { + cout << "KWOptimize : Apres OptimizeGranularizedDataGrid pour Granularite " + << nGranularityIndex << "\t Cout " << dGranularityBestCost << endl; + granularizedOptimizedDataGrid->Write(cout); + } + + // CH IV Begin + // Dans le cas d'un coclustering instances * variables, le cout obtenu + // dGranularityBestCost est le cout de l'antecedent de la meilleure grille avant fusion + // des PV d'un meme cluster L'amelioration de cout doit etre mesuree par rapport au cout + // de la grille post-fusionnee de granularizedOptimizedDataGrid + if (granularizedOptimizedDataGrid->IsVarPartDataGrid()) + { + if (granularizedOptimizedDataGrid->GetInformativeAttributeNumber() > 0 and + optimizationParameters.GetVarPartPostMerge()) + { + dataGridManager.SetSourceDataGrid(granularizedOptimizedDataGrid); + // Creation d'une nouvelle grille avec nouvelle description des PV + + // Calcul de la grille de reference post fusionnee a partir de granularizedDataGrid + dFusionDeltaCost = + dataGridManager.ExportDataGridWithVarPartMergeOptimization( + &granularizedPostMergedOptimizedDataGrid, dataGridCosts); + assert(not granularizedPostMergedOptimizedDataGrid.GetVarPartsShared()); + + // Calcul et verification du cout + dMergedCost = dGranularityBestCost + dFusionDeltaCost; + // Le cout precedent devra etre correct + assert(dMergedCost * (1 - dEpsilon) < + dataGridCosts->ComputeDataGridTotalCost( + &granularizedPostMergedOptimizedDataGrid)); + assert(dataGridCosts->ComputeDataGridTotalCost( + &granularizedPostMergedOptimizedDataGrid) < + dMergedCost * (1 + dEpsilon)); + + if (bDisplayResults) + { + cout << "KWOptimize : Niveau prepartitionnement \t" + << granularizedOptimizedDataGrid->GetInnerAttributes() + ->GetVarPartGranularity() + << "\t Grille avant fusion \t" << dGranularityBestCost + << "\n"; + granularizedOptimizedDataGrid->Write(cout); + } + } + else + dMergedCost = dGranularityBestCost; + + if (dMergedCost < dBestMergedCost - dEpsilon) + { + dBestMergedCost = dMergedCost; + dBestCost = dGranularityBestCost; + + // Memorisation de l'antecedent du nouvel optimum avant post-fusion + dataGridManager.CopyDataGrid(granularizedOptimizedDataGrid, + optimizedDataGrid); + } + + // Cas ou il s'agit de la derniere granularite : on met a jour les infos du + // coclustering + if (bIsLastGranularity) + { + if (bDisplayResults) + cout << "KWOptimize : Mise a jour de la memorisation du " + "coclustering pour la derniere granularite " + << endl; + + if (granularizedPostMergedOptimizedDataGrid + .GetInformativeAttributeNumber() == 0) + HandleOptimizationStep(optimizedDataGrid, &granularizedDataGrid, + true); + + else + { + // La grille source est la grille de reference qui contient la + // partition la plus fine + dataGridManager.SetSourceDataGrid(GetInitialVarPartDataGrid()); + dataGridManager.ExportDataGridWithSingletonVarParts( + &granularizedPostMergedOptimizedDataGrid, + &partitionedReferenceGranularizedPostMergedDataGrid, true); + HandleOptimizationStep( + &granularizedPostMergedOptimizedDataGrid, + &partitionedReferenceGranularizedPostMergedDataGrid, false); + // Nettoyage + partitionedReferenceGranularizedPostMergedDataGrid.DeleteAll(); + } + } + + // Nettoyage + granularizedPostMergedOptimizedDataGrid.DeleteAll(); + } + // CH IV End + + // Sinon : cas coclustering de variables + else + { + // Cas d'amelioration du cout + if (dGranularityBestCost < dBestCost) + { + dBestCost = dGranularityBestCost; + + // Memorisation du nouvel optimum + dataGridManager.CopyDataGrid(granularizedOptimizedDataGrid, + optimizedDataGrid); + + if (bDisplayResults) + { + cout << "KWOptimize :Grille granularizedOptimizedDataGrid" + << endl; + granularizedOptimizedDataGrid->Write(cout); + } + } + // Cas ou il s'agit de la derniere granularite : on met a jour les infos dans le + // cas d'un coclustering + if (bIsLastGranularity) + { + if (bDisplayResults) + cout << "KWOptimize :Mise a jour de la memorisation du " + "coclustering pour la derniere granularite " + << endl; + + HandleOptimizationStep(optimizedDataGrid, &granularizedDataGrid, true); + } + } + + // Nettoyage de la grille granularisee + granularizedDataGrid.DeleteAll(); + + // Nettoyage de la grille optimisee pour cette granularite + delete granularizedOptimizedDataGrid; + granularizedOptimizedDataGrid = NULL; + + // Cas d'un temps limite : mise a jour du temps restant par retrait du temps consacre a + // cette granularite + if (optimizationParameters.GetOptimizationTime() > 0) + { + // L'utilisation de la totalite du temps global alloue (OptimizationTime) peut + // conduire a l'arret du parcours des granularites et nuire a la recherche de la + // grille optimale + if (optimizationParameters.GetOptimizationTime() - dTotalTime > 0) + { + optimizationParameters.SetOptimizationTime( + optimizationParameters.GetOptimizationTime() - (int)dTotalTime); + if (bDisplayResults) + cout << "KWOptimize :Temps restant apres optimisation a la " + "granularite \t" + << nGranularityIndex << "\t " + << optimizationParameters.GetOptimizationTime() << endl; + } + else + { + break; + // Affichage d'un warning pour eventuelle modification de l'optimisation + // time + AddWarning(sTmp + + "All the optimization time has been used but maximum " + "granularity has not been reached:" + + IntToString(nGranularityIndex) + " on " + + IntToString(nGranularityMax) + + ". You could obtain better results with greater " + "optimization time."); + if (bDisplayResults) + cout << "KWOptimize :Totalite du temps alloue ecoule apres la " + "granularite \t" + << nGranularityIndex << endl; + } + } + + // Memorisation du nombre de parties par attribut pour comparaison a l'etape suivante + ivPreviousPartNumber.CopyFrom(&ivCurrentPartNumber); + } + else + { + if (bDisplayGranularities) + cout << "KWOptimize :Granularite " << nGranularityIndex + << " non traitee car identique a la precedente" << endl; + // Nettoyage de la grille granularisee + granularizedDataGrid.DeleteAll(); + } + + nGranularityIndex++; + } + // CH IV Begin + // Cas d'une grille conventionnelle hors coclustering instances * variables + if (not optimizedDataGrid->IsVarPartDataGrid()) + { + // Post-optimisation de la granularite : on attribue a la grille optimale la plus petite + // granularite pour laquelle cette grille est definie + if (nLastExploredGranularity != -1 and + optimizedDataGrid->GetGranularity() > nLastExploredGranularity + 1) + PostOptimizeGranularity(initialDataGrid, optimizedDataGrid, odQuantileBuilders, + nLastExploredGranularity); + } + // CH IV End + + // Nettoyage + odQuantileBuilders.DeleteAll(); + } + // Cas ou la grille terminale n'est pas ameliorable + else + { + // Tri des parties par attribut, pour preparer les affichages de resultats + // ainsi que les resultats de preparation des donnees + optimizedDataGrid->SortAttributeParts(); + } + return dBestCost; +} + +double NEWKWDataGridOptimizer::SimplifyDataGrid(KWDataGrid* optimizedDataGrid) const +{ + KWDataGridManager dataGridManager; + KWDataGridMerger dataGridMerger; + double dSimplifiedGridCost; + + require(optimizedDataGrid != NULL); + + // Cas ou la contrainte est dekja respectee: on garde la grille telle quelle + if (optimizationParameters.GetMaxPartNumber() == 0 or + optimizedDataGrid->ComputeMaxPartNumber() <= optimizationParameters.GetMaxPartNumber()) + { + // Il faut recalculer le cout de la grille initiale + dSimplifiedGridCost = GetDataGridCosts()->ComputeDataGridTotalCost(optimizedDataGrid); + } + // Cas ou il faut simplifier la grille + else + { + // Export de la grille initiale vers un merger de grille + dataGridManager.SetSourceDataGrid(optimizedDataGrid); + dataGridManager.ExportDataGrid(&dataGridMerger); + + // Parametrage des couts et des contraintes + dataGridMerger.SetDataGridCosts(GetDataGridCosts()); + dataGridMerger.SetMaxPartNumber(optimizationParameters.GetMaxPartNumber()); + assert(dataGridMerger.Check()); + + // Post-traitement de simplification + dSimplifiedGridCost = dataGridMerger.Merge(); + + // Memorisation de la solution initiale + dataGridManager.CopyDataGrid(&dataGridMerger, optimizedDataGrid); + } + ensure(fabs(dSimplifiedGridCost - GetDataGridCosts()->ComputeDataGridTotalCost(optimizedDataGrid)) < dEpsilon); + return dSimplifiedGridCost; +} + +double NEWKWDataGridOptimizer::OptimizeGranularizedDataGrid(const KWDataGrid* initialDataGrid, + KWDataGrid* optimizedDataGrid, boolean bIsLastGranularity, + double& dTotalComputeTime) const +{ + boolean bDisplayResults = false; + KWDataGridManager dataGridManager; + KWDataGridMerger dataGridMerger; + KWDataGridPostOptimizer dataGridPostOptimizer; + boolean bOptimizationNeeded; + double dBestCost; + clock_t tBegin; + clock_t tEnd; + ALString sTmp; + + require(initialDataGrid != NULL); + require(initialDataGrid->Check()); + require(optimizedDataGrid != NULL); + + // Debut de suivi des taches + TaskProgression::BeginTask(); + TaskProgression::DisplayMainLabel("Data Grid internal optimization"); + + // Ligne d'entete des messages + DisplayOptimizationHeaderLine(); + + // Initialisations + tBegin = clock(); + dataGridManager.SetSourceDataGrid(initialDataGrid); + dataGridMerger.SetDataGridCosts(dataGridCosts); + dataGridPostOptimizer.SetDataGridCosts(dataGridCosts); + + // Initialisation du meilleur cout + dBestCost = dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid); + + if (bDisplayResults) + { + cout << "Debut OptimizeGranularizedDataGrid " << endl; + cout << "Grille initiale" << endl; + initialDataGrid->Write(cout); + cout << "Grille optimale" << endl; + optimizedDataGrid->Write(cout); + cout << "Granularite courante \t " << initialDataGrid->GetGranularity() + << endl; //<< "\t Meilleure granu \t " << GetGranularity()->GetBestGranularity() << endl; + cout << " Cout grille optimale a la granularite courante " << dBestCost << endl; + } + + // On determine si on peut potentiellement faire mieux que la grille terminale + bOptimizationNeeded = true; + if ((initialDataGrid->GetTargetAttribute() == NULL and initialDataGrid->GetTargetValueNumber() == 1) or + (initialDataGrid->GetTargetAttribute() == NULL and initialDataGrid->GetInformativeAttributeNumber() == 0) or + (initialDataGrid->GetTargetAttribute() != NULL and + initialDataGrid->GetTargetAttribute()->GetPartNumber() <= 1) or + (initialDataGrid->GetTargetAttribute() != NULL and initialDataGrid->GetInformativeAttributeNumber() <= 1)) + bOptimizationNeeded = false; + + // Initialisation univariee + if (bOptimizationNeeded and optimizationParameters.GetUnivariateInitialization() and + initialDataGrid->GetTargetValueNumber() > 0 + // on ne fait pas d'univarie si un seul attribut + and initialDataGrid->GetAttributeNumber() > 1) + { + // Avec prise en compte de la granularite + dBestCost = + OptimizeWithBestUnivariatePartitionForCurrentGranularity(initialDataGrid, optimizedDataGrid); + + // Recherche d'une amelioration par croisement des partitions univariees + // Integre la granularite + dBestCost = OptimizeWithMultipleUnivariatePartitions(initialDataGrid, optimizedDataGrid); + } + + // Affichage + if (bDisplayResults) + { + cout << "Grille optimisee avant VNS" << endl; + optimizedDataGrid->Write(cout); + } + + // Optimisation a partir d'une grille initiale complete si algorithme glouton + if (bOptimizationNeeded) + { + // Optimisation avec algorithme greedy + if (optimizationParameters.GetOptimizationAlgorithm() == "Greedy") + dBestCost = GreedyOptimize(initialDataGrid, optimizedDataGrid); + // Optimisation avec algorithme multi-start + else if (optimizationParameters.GetOptimizationAlgorithm() == "MultiStart") + dBestCost = MultiStartOptimize(initialDataGrid, optimizedDataGrid); + // Optimisation avec algorithme VNS + else if (optimizationParameters.GetOptimizationAlgorithm() == "VNS") + dBestCost = VNSOptimize(initialDataGrid, optimizedDataGrid, bIsLastGranularity); + } + + // Tri des parties par attribut, pour preparer les affichages de resultats + // ainsi que les resultats de preparation des donnees + optimizedDataGrid->SortAttributeParts(); + + // Affichage de la grille finale avec ses couts + if (optimizationParameters.GetDisplayDetails() and bDisplayResults) + { + // Grille optimisee + cout << "Optimized grid\n"; + if (optimizedDataGrid->GetAttributeNumber() == 2) + optimizedDataGrid->WriteCrossTableStats(cout, 0); + dataGridCosts->WriteDataGridAllCosts(optimizedDataGrid, cout); + cout << endl; + + // Grille finale + cout << *optimizedDataGrid << endl; + } + // Grille finale + if (bDisplayResults) + { + cout << "Granularite courante \t " << optimizedDataGrid->GetGranularity() << "\t Meilleure granu \t " + << endl; // GetGranularity()->GetBestGranularity() << endl; + cout << " Cout meilleure grille " << dBestCost << endl; + } + + // Affichage du temps de calcul + tEnd = clock(); + dTotalComputeTime = (double)(tEnd - tBegin) / CLOCKS_PER_SEC; + if (optimizationParameters.GetDisplayDetails()) + initialDataGrid->AddMessage(sTmp + + "Data grid optimization time: " + SecondsToString((int)dTotalComputeTime)); + + // Fin de suivi des taches + TaskProgression::EndTask(); + + // Retour du meilleur cout de codage + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid) - dBestCost) < dEpsilon); + return dBestCost; +} + +void NEWKWDataGridOptimizer::HandleOptimizationStep(const KWDataGrid* optimizedDataGrid, + const KWDataGrid* initialGranularizedDataGrid, + boolean bIsLastSaving) const +{ + // Integration de la granularite + if (attributeSubsetStatsHandler != NULL) + attributeSubsetStatsHandler->HandleOptimizationStep(optimizedDataGrid, initialGranularizedDataGrid, + bIsLastSaving); +} + +void NEWKWDataGridOptimizer::SetAttributeSubsetStats(const KWAttributeSubsetStats* attributeSubsetStats) +{ + attributeSubsetStatsHandler = attributeSubsetStats; +} + +const KWAttributeSubsetStats* NEWKWDataGridOptimizer::GetAttributeSubsetStats() +{ + return attributeSubsetStatsHandler; +} + +void NEWKWDataGridOptimizer::PostOptimizeGranularity(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid, + ObjectDictionary& odQuantileBuilders, + int nLastExploredGranularity) const +{ + int nCurrentGranularity; + int nBestGranularity; + int nGranularityPartileNumber; + int nGranularisationIntervalIndex; + int nPartitionCumulatedFrequency; + int nGranularisationCumulatedFrequency; + int nAttributeIndex; + boolean bIncompatibleGranularity; + boolean bLastExploredGranularity; + KWDGAttribute* attribute; + KWDGAttribute* initialAttribute; + KWDGPart* part; + KWQuantileGroupBuilder* quantileGroupBuilder; + KWQuantileIntervalBuilder* quantileIntervalBuilder; + ObjectArray oaInitialParts; + int nCatchAllFirstModalityIndex; + int nCatchAllModalityIndex; + KWDGPart* catchAllPart; + IntVector ivPartileNumber; + IntVector ivBestPartileNumber; + + // Initialisation + bLastExploredGranularity = false; + bIncompatibleGranularity = false; + nCurrentGranularity = optimizedDataGrid->GetGranularity() - 1; + nBestGranularity = optimizedDataGrid->GetGranularity(); + + require(nBestGranularity > nLastExploredGranularity + 1); + + // Initialisation du nombre de partiles par attribut + nAttributeIndex = 0; + while (nAttributeIndex < optimizedDataGrid->GetAttributeNumber()) + { + // Extraction de l'attribut courant + attribute = optimizedDataGrid->GetAttributeAt(nAttributeIndex); + + ivBestPartileNumber.Add(attribute->GetGranularizedValueNumber()); + nAttributeIndex++; + } + + // Boucle descendante sur les granularites jusqu'a rencontrer l'avant derniere granularite exploree ou une + // granularite incompatible avec la partition + while (not bLastExploredGranularity and not bIncompatibleGranularity) + { + nGranularityPartileNumber = (int)pow(2.0, nCurrentGranularity); + + // Initialisation de l'attribut etudie + nAttributeIndex = 0; + ivPartileNumber.SetSize(0); + + // Boucle sur les attributs tant que l'on n'a pas rencontre un attribut dont la granularisation est + // incompatible avec la grille + while (nAttributeIndex < optimizedDataGrid->GetAttributeNumber() and not bIncompatibleGranularity) + { + // Extraction de l'attribut courant + attribute = optimizedDataGrid->GetAttributeAt(nAttributeIndex); + + // Reinitialisation du tableau des parties + oaInitialParts.SetSize(0); + + // Cas ou le nombre de partiles theorique de cette granularite est superieur ou egal a la taille + // de la partition de l'attribut courant + if (nGranularityPartileNumber >= attribute->GetPartNumber()) + { + // Cas d'un attribut continu + if (attribute->GetAttributeType() == KWType::Continuous) + { + quantileIntervalBuilder = + cast(KWQuantileIntervalBuilder*, + odQuantileBuilders.Lookup(attribute->GetAttributeName())); + // Memorisation du nombre de partiles (theorique pour attribut numerique, + // effectif pour attribut categoriel) + ivPartileNumber.Add(nGranularityPartileNumber); + nGranularityPartileNumber = + quantileIntervalBuilder->ComputeQuantiles(nGranularityPartileNumber); + + // Cas ou le nombre reel de cette granularisation est superieur ou egal a la + // taille de la partition + if (nGranularityPartileNumber >= attribute->GetPartNumber()) + { + part = attribute->GetHeadPart(); + nPartitionCumulatedFrequency = part->GetPartFrequency(); + + // Initialisation + nGranularisationIntervalIndex = 0; + nGranularisationCumulatedFrequency = + quantileIntervalBuilder->GetIntervalLastInstanceIndexAt( + nGranularisationIntervalIndex) + + 1; + + // Parcours des intervalles de la partition optimale et des intervalles + // de la granularisation tant qu'il y a compatibilite + while (not bIncompatibleGranularity and part != NULL) + { + while (nPartitionCumulatedFrequency > + nGranularisationCumulatedFrequency) + { + nGranularisationIntervalIndex++; + nGranularisationCumulatedFrequency = + quantileIntervalBuilder + ->GetIntervalLastInstanceIndexAt( + nGranularisationIntervalIndex) + + 1; + } + + if (nPartitionCumulatedFrequency < + quantileIntervalBuilder->GetIntervalLastInstanceIndexAt( + nGranularisationIntervalIndex) + + 1) + bIncompatibleGranularity = true; + // Partie suivante + attribute->GetNextPart(part); + if (part != NULL) + nPartitionCumulatedFrequency += + part->GetPartFrequency(); + } + } + else + bIncompatibleGranularity = true; + } + // Cas d'un attribut categoriel + else + { + quantileGroupBuilder = + cast(KWQuantileGroupBuilder*, + odQuantileBuilders.Lookup(attribute->GetAttributeName())); + + nGranularityPartileNumber = + quantileGroupBuilder->ComputeQuantiles(nGranularityPartileNumber); + // Memorisation du nombre de partiles (theorique pour attribut numerique, + // effectif pour attribut categoriel) + ivPartileNumber.Add(nGranularityPartileNumber); + + // Cas ou le nombre reel de cette granularisation est superieur ou egal a la + // taille de la partition + if (nGranularityPartileNumber >= attribute->GetPartNumber()) + { + attribute->BuildIndexingStructure(); + + // Export des parties de l'attribut initial + initialAttribute = + initialDataGrid->SearchAttribute(attribute->GetAttributeName()); + initialAttribute->ExportParts(&oaInitialParts); + // Extraction de l'index de la 1ere modalite du fourre-tout. Le + // fourre-tout est le dernier groupe du quantileBuilder + nCatchAllFirstModalityIndex = + quantileGroupBuilder->GetGroupFirstValueIndexAt( + nGranularityPartileNumber - 1); + + // Extraction de la partie a ajouter dans le groupe + catchAllPart = + cast(KWDGPart*, oaInitialParts.GetAt(nCatchAllFirstModalityIndex)); + + // Parcours des autres modalites du fourre-tout pour verifier qu'elles + // sont dans le meme groupe que la premiere modalite du fourre-tout + nCatchAllModalityIndex = nCatchAllFirstModalityIndex + 1; + while (not bIncompatibleGranularity and + nCatchAllModalityIndex <= + quantileGroupBuilder->GetGroupLastValueIndexAt( + nGranularityPartileNumber - 1)) + { + part = cast(KWDGPart*, + oaInitialParts.GetAt(nCatchAllModalityIndex)); + if (part != catchAllPart) + bIncompatibleGranularity = true; + nCatchAllModalityIndex++; + } + attribute->DeleteIndexingStructure(); + } + else + bIncompatibleGranularity = true; + } + } + else + bIncompatibleGranularity = true; + + nAttributeIndex++; + } + + // Cas d'une granularite compatible : memorisation de la granularite et du nombre de partiles par + // attribut de la grille + if (not bIncompatibleGranularity) + { + nBestGranularity = nCurrentGranularity; + ivBestPartileNumber.CopyFrom(&ivPartileNumber); + } + + // Prochaine granularite + nCurrentGranularity--; + bLastExploredGranularity = (nCurrentGranularity == nLastExploredGranularity); + } + + // Mise a jour de la granularite de la grille et du nombre de partiles pour chaque attribut + optimizedDataGrid->SetGranularity(nBestGranularity); + nAttributeIndex = 0; + while (nAttributeIndex < optimizedDataGrid->GetAttributeNumber()) + { + // Extraction de l'attribut courant + attribute = optimizedDataGrid->GetAttributeAt(nAttributeIndex); + + attribute->SetGranularizedValueNumber(ivBestPartileNumber.GetAt(nAttributeIndex)); + nAttributeIndex++; + } +} + +double NEWKWDataGridOptimizer::InitializeWithTerminalDataGrid(const KWDataGrid* initialDataGrid, + KWDataGrid* optimizedDataGrid) const +{ + double dBestCost; + KWDataGridManager dataGridManager; + KWDataGrid terminalDataGrid; + + // Initialisations + dataGridManager.SetSourceDataGrid(initialDataGrid); + + // Construction d'une grille terminale pour la solution initiale + dataGridManager.ExportTerminalDataGrid(&terminalDataGrid); + dBestCost = dataGridCosts->GetTotalDefaultCost(); + + // Memorisation de la meilleure solution initiale + if (bCleanNonInformativeVariables) + dataGridManager.CopyInformativeDataGrid(&terminalDataGrid, optimizedDataGrid); + else + dataGridManager.CopyDataGrid(&terminalDataGrid, optimizedDataGrid); + + // Affichage du resulat (ici: grilles initiale et optimisee sont confondues) + DisplayOptimizationDetails(optimizedDataGrid, false); + DisplayOptimizationDetails(optimizedDataGrid, true); + + // Retour du cout + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid) - dBestCost) < dEpsilon); + return dBestCost; +} + +double +NEWKWDataGridOptimizer::OptimizeWithBestUnivariatePartitionForCurrentGranularity(const KWDataGrid* initialDataGrid, + KWDataGrid* optimizedDataGrid) const +{ + KWAttributeStats* attributeStats; + KWDataGridManager dataGridManager; + KWDataGrid univariateDataGrid; + KWDGAttribute* initialAttribute; + KWDGAttribute* targetAttribute; + double dBestCost; + double dCost; + int nAttribute; + boolean bEvaluated; + boolean bImproved; + boolean bDisplayResults = false; + + require(initialDataGrid != NULL); + require(optimizedDataGrid != NULL); + require(initialDataGrid->GetTargetValueNumber() > 0); + + // Initialisations + dBestCost = dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid); + dataGridManager.SetSourceDataGrid(initialDataGrid); + + if (bDisplayResults) + cout << " OptimizeWithBestUnivariate cout initial " << dBestCost << endl; + + // Retour si pas de statistiques univariees disponibles + if (classStats == NULL) + return dBestCost; + + // Recherche des grilles univariees deduites des statistiques univariees + bImproved = false; + + // Parcours des attributs + for (nAttribute = 0; nAttribute < initialDataGrid->GetAttributeNumber(); nAttribute++) + { + initialAttribute = initialDataGrid->GetAttributeAt(nAttribute); + + // Initialisation de la grille univariee (uniquement si plus de une partie dans la partition) + univariateDataGrid.DeleteAll(); + bEvaluated = false; + + // Extraction de la partition stockee dans l'attributeStats + attributeStats = classStats->LookupAttributeStats(initialAttribute->GetAttributeName()); + bEvaluated = attributeStats->GetLevel() > 0; + + // Cas d'un attribut informatif + if (bEvaluated) + { + // Cas ou la granularite de la meilleure partition univarie de l'attribut ne correspond pas + // a la granularite courante de la grille + if (attributeStats->GetPreparedDataGridStats()->GetGranularity() != + initialDataGrid->GetGranularity()) + { + dataGridManager.BuildUnivariateDataGridFromGranularizedPartition( + &univariateDataGrid, nAttribute, classStats); + bEvaluated = univariateDataGrid.GetAttributeAt(0)->GetPartNumber() > 1; + } + + // Cas ou la granularite de la meilleure partition univariee de l'attribut correspond + // a la granularite courante de la grille + else + { + dataGridManager.BuildUnivariateDataGridFromAttributeStats(&univariateDataGrid, + attributeStats); + + // Transfert du parametrage du fourre-tout + targetAttribute = univariateDataGrid.GetAttributeAt(0); + targetAttribute->InitializeCatchAllValueSet(initialAttribute->GetCatchAllValueSet()); + } + } + + // Evaluation si la grille univariee a ete construite + if (bEvaluated) + { + // Evaluation de la grille + dCost = dataGridCosts->ComputeDataGridTotalCost(&univariateDataGrid); + if (bDisplayResults) + cout << " APRES Cout de la grille univariee " << dCost << endl; + + // Affichage du resulat (ici: grilles initiale et optimisee sont confondues) + DisplayOptimizationDetails(&univariateDataGrid, false); + DisplayOptimizationDetails(&univariateDataGrid, true); + + // Memorisation de la meilleure solution + if (dCost < dBestCost - dEpsilon) + { + dBestCost = dCost; + dataGridManager.CopyDataGrid(&univariateDataGrid, optimizedDataGrid); + bImproved = true; + } + } + } + + // Retour du cout + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid) - dBestCost) == 0 or bImproved); + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid) - dBestCost) < dEpsilon); + return dBestCost; +} + +double NEWKWDataGridOptimizer::OptimizeWithMultipleUnivariatePartitions(const KWDataGrid* initialDataGrid, + KWDataGrid* optimizedDataGrid) const +{ + double dBestCost; + double dCost; + KWDataGridManager dataGridManager; + KWDataGridMerger multivariateDataGrid; + KWDataGridPostOptimizer dataGridPostOptimizer; + boolean bOk; + boolean bDisplayResults = false; + + require(initialDataGrid != NULL); + require(optimizedDataGrid != NULL); + require(initialDataGrid->GetTargetValueNumber() > 0); + + // Initialisations + dBestCost = dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid); + dataGridManager.SetSourceDataGrid(initialDataGrid); + multivariateDataGrid.SetDataGridCosts(dataGridCosts); + dataGridPostOptimizer.SetDataGridCosts(dataGridCosts); + + if (bDisplayResults) + { + cout << " OptimizeWithMultipleUnivariatePartitions :: dBestCost initial " << dBestCost << endl; + cout << " Grille initiale " << endl; + cout << *optimizedDataGrid; + } + + // Retour si pas assez de statistiques univariees disponibles + if (classStats == NULL or classStats->GetInformativeAttributeNumber() <= 1) + return dBestCost; + + // Construction d'une grille par croisement des partition univariee + // Traitement avec calcul des partitions univariees pour cette granularite + bOk = dataGridManager.BuildDataGridFromUnivariateProduct(&multivariateDataGrid, classStats); + + if (not bOk) + return dBestCost; + + // Affichage du cout initial + DisplayOptimizationDetails(&multivariateDataGrid, false); + + // Cout initial si aucune optimisation + dCost = DBL_MAX; + if (not optimizationParameters.GetOptimize() and not optimizationParameters.GetPreOptimize() and + not optimizationParameters.GetPostOptimize()) + dCost = dataGridCosts->ComputeDataGridTotalCost(&multivariateDataGrid); + + if (bDisplayResults) + { + dCost = dataGridCosts->ComputeDataGridTotalCost(&multivariateDataGrid); + cout << " OptimizeWithMultipleUnivariatePartitions :: dBestCost multivarie initial " << dCost << endl; + cout << " Grille initiale " << endl; + cout << multivariateDataGrid; + } + + // Pre-optimisation de la grille + if (optimizationParameters.GetPreOptimize()) + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, &multivariateDataGrid, false); + + // Optimisation par fusion des groupes + if (optimizationParameters.GetOptimize()) + dCost = multivariateDataGrid.Merge(); + + // Post-optimisation de la grille + if (optimizationParameters.GetPostOptimize()) + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, &multivariateDataGrid, true); + + if (bDisplayResults) + { + cout << " OptimizeWithMultipleUnivariatePartitions :: dBestCost multivarie optimise " << dCost << endl; + cout << " Grille multivariee optimisee " << endl; + cout << multivariateDataGrid; + } + + // Affichage du cout final + DisplayOptimizationDetails(&multivariateDataGrid, true); + + // Memorisation de la meilleure solution + if (dCost < dBestCost - dEpsilon) + { + dBestCost = dCost; + if (bCleanNonInformativeVariables) + dataGridManager.CopyInformativeDataGrid(&multivariateDataGrid, optimizedDataGrid); + else + dataGridManager.CopyDataGrid(&multivariateDataGrid, optimizedDataGrid); + } + + // Retour du cout + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid) - dBestCost) < dEpsilon); + return dBestCost; +} + +double NEWKWDataGridOptimizer::GreedyOptimize(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid) const +{ + KWDataGridManager dataGridManager; + KWDataGridMerger dataGridMerger; + KWDataGridPostOptimizer dataGridPostOptimizer; + double dBestCost; + double dCost; + + // Initialisations + dBestCost = dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid); + dataGridManager.SetSourceDataGrid(initialDataGrid); + dataGridMerger.SetDataGridCosts(dataGridCosts); + dataGridPostOptimizer.SetDataGridCosts(dataGridCosts); + + // Export complet + dataGridManager.ExportDataGrid(&dataGridMerger); + + // Affichage du cout initial + DisplayOptimizationDetails(&dataGridMerger, false); + + // Cout initial si aucune optimisation + dCost = DBL_MAX; + if (not optimizationParameters.GetOptimize() and not optimizationParameters.GetPreOptimize() and + not optimizationParameters.GetPostOptimize()) + dCost = dataGridCosts->ComputeDataGridTotalCost(&dataGridMerger); + + // Pre-optimisation de la grille + if (optimizationParameters.GetPreOptimize()) + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, &dataGridMerger, false); + + // Optimisation par fusion des groupes + if (optimizationParameters.GetOptimize()) + dCost = dataGridMerger.Merge(); + + // Post-optimisation de la grille + if (optimizationParameters.GetPostOptimize()) + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, &dataGridMerger, true); + + // Affichage du cout final + DisplayOptimizationDetails(&dataGridMerger, true); + + // Memorisation de la meilleure solution + if (dCost < dBestCost - dEpsilon) + { + dBestCost = dCost; + if (bCleanNonInformativeVariables) + dataGridManager.CopyInformativeDataGrid(&dataGridMerger, optimizedDataGrid); + else + dataGridManager.CopyDataGrid(&dataGridMerger, optimizedDataGrid); + } + + // Retour du cout + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid) - dBestCost) < dEpsilon); + return dBestCost; +} + +double NEWKWDataGridOptimizer::MultiStartOptimize(const KWDataGrid* initialDataGrid, + KWDataGrid* optimizedDataGrid) const +{ + KWDataGridManager dataGridManager; + KWDataGridMerger dataGridMerger; + KWDataGridPostOptimizer dataGridPostOptimizer; + double dBestCost; + double dCost; + const int nDisplayedTry = -1; + int nTryNumber; + int nTry; + int nAttributeNumber; + int nPartNumber; + Timer timerOptimization; + + // Initialisations + dBestCost = dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid); + dataGridManager.SetSourceDataGrid(initialDataGrid); + dataGridMerger.SetDataGridCosts(dataGridCosts); + dataGridPostOptimizer.SetDataGridCosts(dataGridCosts); + + // Multi-start, partant de grilles aleatoire de taille "raisonnable" + nTryNumber = (int)pow(2.0, optimizationParameters.GetOptimizationLevel()); + for (nTry = 0; nTry < nTryNumber; nTry++) + { + timerOptimization.Start(); + + // Reinitialisation + dataGridMerger.DeleteAll(); + + // Specification du nombre d'attributs et de parties + nAttributeNumber = 1 + RandomInt((int)(log(initialDataGrid->GetGridFrequency() * 1.0) / log(2.0))); + if (nAttributeNumber < 2) + nAttributeNumber = 2; + if (nAttributeNumber > initialDataGrid->GetAttributeNumber()) + nAttributeNumber = initialDataGrid->GetAttributeNumber(); + nPartNumber = + 2 + 2 * RandomInt((int)pow(initialDataGrid->GetGridFrequency() * 1.0, 1.0 / nAttributeNumber)); + if (nPartNumber > initialDataGrid->GetGridFrequency()) + nPartNumber = initialDataGrid->GetGridFrequency(); + + // Export d'une grille aleatoire + dataGridManager.ExportRandomAttributes(&dataGridMerger, nAttributeNumber); + dataGridManager.ExportRandomParts(&dataGridMerger, nPartNumber); + dataGridManager.ExportCells(&dataGridMerger); + + // Affichage du cout initial + DisplayOptimizationDetails(&dataGridMerger, false); + + // Cout initial si aucune optimisation + dCost = DBL_MAX; + if (not optimizationParameters.GetOptimize() and not optimizationParameters.GetPreOptimize() and + not optimizationParameters.GetPostOptimize()) + dCost = dataGridCosts->ComputeDataGridTotalCost(&dataGridMerger); + + // Pre-optimisation de la grille + if (optimizationParameters.GetPreOptimize()) + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, &dataGridMerger, false); + + // Optimisation par fusion des groupes + if (optimizationParameters.GetOptimize()) + dCost = dataGridMerger.Merge(); + + // Post-optimisation de la grille + if (optimizationParameters.GetPostOptimize()) + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, &dataGridMerger, true); + + // Affichage du cout final + DisplayOptimizationDetails(&dataGridMerger, true); + + // Affichage d'une grille particuliere + if (optimizationParameters.GetDisplayDetails() and nTry == nDisplayedTry) + { + cout << dataGridMerger << endl; + if (dataGridMerger.GetAttributeNumber() == 2) + dataGridMerger.WriteCrossTableStats(cout, 0); + dataGridMerger.GetDataGridCosts()->WriteDataGridAllCosts(&dataGridMerger, cout); + cout << endl; + break; + } + + // Memorisation de la meilleure solution + if (dCost < dBestCost - dEpsilon) + { + dBestCost = dCost; + if (bCleanNonInformativeVariables) + dataGridManager.CopyInformativeDataGrid(&dataGridMerger, optimizedDataGrid); + else + dataGridManager.CopyDataGrid(&dataGridMerger, optimizedDataGrid); + + // Gestion de la meilleure solution + HandleOptimizationStep(optimizedDataGrid, NULL, false); + } + + // Arret si contrainte de temps depassee + timerOptimization.Stop(); + if (optimizationParameters.GetOptimizationTime() > 0 and + timerOptimization.GetElapsedTime() > optimizationParameters.GetOptimizationTime()) + break; + } + + // Retour du cout + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid) - dBestCost) < dEpsilon); + return dBestCost; +} + +double NEWKWDataGridOptimizer::VNSOptimize(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid, + boolean bIsLastGranularity) const +{ + NEWKWDataGridVNSOptimizer dataGridVNSOptimizer; + double dBestCost; + + // Optimisation VNS + dataGridVNSOptimizer.SetDataGridCosts(dataGridCosts); + dataGridVNSOptimizer.GetParameters()->CopyFrom(&optimizationParameters); + dataGridVNSOptimizer.SetDataGridOptimizer(this); + + // Cas non supervise (co-clustering) + // Mise a true du parametre bSlightOptimizationMode pour les granularites intermediaires + if (initialDataGrid->GetTargetValueNumber() == 0 and initialDataGrid->GetTargetAttribute() == NULL and + not bIsLastGranularity) + dataGridVNSOptimizer.SetSlightOptimizationMode(true); + // Sinon : cas supervise ou derniere granularite en co-clustering + else + dataGridVNSOptimizer.SetSlightOptimizationMode(false); + + dBestCost = dataGridVNSOptimizer.OptimizeDataGrid(initialDataGrid, optimizedDataGrid); + + // Retour du cout + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid) - dBestCost) < dEpsilon); + return dBestCost; +} + +void NEWKWDataGridOptimizer::DisplayOptimizationHeaderLine() const +{ + if (optimizationParameters.GetDisplayDetails()) + { + // Lignes d'entete + cout << "Initial\t\t\t\tFinal\t\t\t\t\n"; + cout << "Att. number\tPart number\tCell number\tCost\t"; + cout << "Att. number\tPart number\tCell number\tCost\n"; + } +} + +void NEWKWDataGridOptimizer::DisplayOptimizationDetails(const KWDataGrid* optimizedDataGrid, boolean bOptimized) const +{ + if (optimizationParameters.GetDisplayDetails()) + { + // Affichage des caracteristiques de la grille terminale + cout << optimizedDataGrid->GetAttributeNumber() << "\t" << optimizedDataGrid->GetTotalPartNumber() + << "\t" << optimizedDataGrid->GetCellNumber() << "\t" + << dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid); + if (not bOptimized) + cout << "\t"; + else + cout << "\n"; + cout << flush; + } +} + +Profiler NEWKWDataGridOptimizer::profiler; + +////////////////////////////////////////////////////////////////////////////////// +// Classe NEWKWDataGridVNSOptimizer + +NEWKWDataGridVNSOptimizer::NEWKWDataGridVNSOptimizer() +{ + dataGridCosts = NULL; + bCleanNonInformativeVariables = false; + nVNSIteration = 0; + nVNSLevel = 0; + nVNSMaxLevel = 0; + dVNSNeighbourhoodSize = 0; + dEpsilon = 1e-6; + dataGridOptimizer = NULL; + bSlightOptimizationMode = false; +} + +NEWKWDataGridVNSOptimizer::~NEWKWDataGridVNSOptimizer() {} + +void NEWKWDataGridVNSOptimizer::SetDataGridCosts(const KWDataGridCosts* kwdgcCosts) +{ + dataGridCosts = kwdgcCosts; +} + +const KWDataGridCosts* NEWKWDataGridVNSOptimizer::GetDataGridCosts() const +{ + return dataGridCosts; +} + +KWDataGridOptimizerParameters* NEWKWDataGridVNSOptimizer::GetParameters() +{ + return &optimizationParameters; +} + +void NEWKWDataGridVNSOptimizer::SetSlightOptimizationMode(boolean bValue) +{ + bSlightOptimizationMode = bValue; +} + +const boolean NEWKWDataGridVNSOptimizer::GetSlightOptimizationMode() const +{ + return bSlightOptimizationMode; +} + +double NEWKWDataGridVNSOptimizer::OptimizeDataGrid(const KWDataGrid* initialDataGrid, + KWDataGrid* optimizedDataGrid) const +{ + int nMaxLevel; + + require(initialDataGrid != NULL); + require(initialDataGrid->Check()); + require(optimizedDataGrid != NULL); + + // On prend au minimum un niveau max de 1 + nMaxLevel = optimizationParameters.GetOptimizationLevel(); + if (nMaxLevel <= 0) + nMaxLevel = 1; + + // Optimisation + return IterativeVNSOptimizeDataGrid(initialDataGrid, nMaxLevel, optimizedDataGrid); +} + +void NEWKWDataGridVNSOptimizer::SetDataGridOptimizer(const NEWKWDataGridOptimizer* optimizer) +{ + dataGridOptimizer = optimizer; +} + +const NEWKWDataGridOptimizer* NEWKWDataGridVNSOptimizer::GetDataGridOptimizer() const +{ + return dataGridOptimizer; +} + +double NEWKWDataGridVNSOptimizer::IterativeVNSOptimizeDataGrid(const KWDataGrid* initialDataGrid, int nMaxLevel, + KWDataGrid* optimizedDataGrid) const +{ + int nLevel; + double dCost; + double dBestCost; + KWDataGridManager dataGridManager; + KWDataGrid currentDataGrid; + double dMaxNeighbourhoodSize; + double dMinNeighbourhoodSize; + double dDecreaseFactor; + int nIndexNumber; + boolean bDisplayResults = false; + // CH IV Begin + double dMergedCost; + double dBestMergedCost; + ALString sLabel; + // CH IV End + + // On ne reverifie pas les precondition de la methode publique + require(1 <= nMaxLevel); + + // Demarrage du timer + timerVNS.Reset(); + timerVNS.Start(); + + // Ligne d'entete des messages + DisplayOptimizationHeaderLine(); + + // Initialisations + dataGridManager.SetSourceDataGrid(initialDataGrid); + dBestCost = dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid); + // CH IV Begin + dBestMergedCost = dBestCost; + // CH IV End + + // Calcul d'une taille minimale de voisinnage, relative a la taille de la base + // On souhaite impliquer au minimum 3 nouvelles parties par attribut + dMinNeighbourhoodSize = 3.0 / (3 + initialDataGrid->GetGridFrequency()); + + // La taile max est de 1 + dMaxNeighbourhoodSize = 1; + + // Appel de VNS en augmentant le nombre de voisinnages d'un facteur 2 chaque fois + nVNSIteration = 0; + for (nLevel = 0; nLevel < nMaxLevel; nLevel++) + { + if (bDisplayResults) + cout << "IterativeVNSOptimizeDataGrid :: nLevel \t" << nLevel << endl; + + // Recopie de la meilleure solution dans une solution de travail courante + // CH IV Begin + dataGridManager.SetSourceDataGrid(initialDataGrid); + // CH IV End + dataGridManager.CopyDataGrid(optimizedDataGrid, ¤tDataGrid); + dCost = dataGridCosts->ComputeDataGridTotalCost(¤tDataGrid); + + // Calcul du nombre d'index + nIndexNumber = int(pow(2.0, nLevel)); + + // Calcul du taux de decroissance des taille de voisinnage de facon a obtenir + // une taille minimale de voisinnage suffisante + dDecreaseFactor = 1.0 / pow(dMinNeighbourhoodSize, 1.0 / (nIndexNumber + 1)); + + // Parametrage du profiling + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("VNS optimize"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString( + "Is VarPart", BooleanToString(currentDataGrid.IsVarPartDataGrid())); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Level", IntToString(nLevel)); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Index number", IntToString(nIndexNumber)); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Decrease factor", + DoubleToString(dDecreaseFactor)); + + // Optimisation a partir de la nouvelle solution + // CH IV Begin + // Cas d'un coclustering de variables + if (not currentDataGrid.IsVarPartDataGrid()) + { + dCost = VNSOptimizeDataGrid(initialDataGrid, dDecreaseFactor, 0, nIndexNumber, ¤tDataGrid, + dCost); + if (dCost < dBestCost - dEpsilon) + { + dBestCost = dCost; + if (bCleanNonInformativeVariables) + dataGridManager.CopyInformativeDataGrid(¤tDataGrid, optimizedDataGrid); + else + dataGridManager.CopyDataGrid(¤tDataGrid, optimizedDataGrid); + } + } + // Sinon, cas d'une grille VarPart avec attribut de type VarPart + else + { + // On distingue le cout dMergedCost de la meilleure grille et le cout dCost de l'antecedent de + // la meilleure grille sans fusion des PV adjacentes dans un meme cluster + dMergedCost = dCost; + dCost = VNSOptimizeVarPartDataGrid(initialDataGrid, dDecreaseFactor, 0, nIndexNumber, + ¤tDataGrid, dCost, dMergedCost); + if (dMergedCost < dBestMergedCost - dEpsilon) + { + dBestCost = dCost; + dBestMergedCost = dMergedCost; + + if (bCleanNonInformativeVariables) + dataGridManager.CopyInformativeDataGrid(¤tDataGrid, optimizedDataGrid); + else + dataGridManager.CopyDataGrid(¤tDataGrid, optimizedDataGrid); + } + } + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("VNS optimize"); + // CH IV End + + // Test de fin de tache + if (TaskProgression::IsInterruptionRequested()) + break; + + // Test si depassement de temps + if (IsOptimizationTimeElapsed()) + break; + + // Test cas d'une unique iteration + if (GetSlightOptimizationMode()) + break; + } + assert(dBestCost < DBL_MAX); + + // Arret du timer + timerVNS.Stop(); + timerVNS.Reset(); + + ensure(fabs(dBestCost - dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid)) < dEpsilon); + return dBestCost; +} + +double NEWKWDataGridVNSOptimizer::VNSOptimizeDataGrid(const KWDataGrid* initialDataGrid, double dDecreaseFactor, + int nMinIndex, int nMaxIndex, KWDataGrid* optimizedDataGrid, + double dOptimizedDataGridCost) const +{ + double dBestCost; + double dCost; + KWDataGridManager dataGridManager; + KWDataGridMerger neighbourDataGrid; + int nIndex; + double dNeighbourhoodSize; + + // On ne reverifie pas les precondition de la methode publique + require(dDecreaseFactor > 1); + require(0 <= nMinIndex); + require(nMinIndex <= nMaxIndex); + require(fabs(dOptimizedDataGridCost - dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid)) < dEpsilon); + + // Initialisations + dataGridManager.SetSourceDataGrid(initialDataGrid); + neighbourDataGrid.SetDataGridCosts(dataGridCosts); + dBestCost = dOptimizedDataGridCost; + + // On optimise tant qu'on ne depasse pas la taille max de voisinnage + nVNSMaxLevel = nMaxIndex; + nIndex = nMinIndex; + while (nIndex <= nMaxIndex) + { + nVNSLevel = nIndex; + dNeighbourhoodSize = pow(1.0 / dDecreaseFactor, nIndex); + dVNSNeighbourhoodSize = dNeighbourhoodSize; + + // Generation d'une solution dans un voisinnage de la meilleure solution + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Generate neighbour solution"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Neighbourhood size", + DoubleToString(dNeighbourhoodSize)); + GenerateNeighbourSolution(initialDataGrid, optimizedDataGrid, dNeighbourhoodSize, &neighbourDataGrid); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Generate neighbour solution"); + + // Parametrage du profiling pour l'optimisation + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Optimize solution"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Index", IntToString(nIndex)); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("VNS neighbourhood size", + DoubleToString(dVNSNeighbourhoodSize)); + + // Optimisation de cette solution + dCost = OptimizeSolution(initialDataGrid, &neighbourDataGrid); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Optimize solution"); + + // Si amelioration: on la memorise + if (dCost < dBestCost - dEpsilon) + { + dBestCost = dCost; + + // Sauvegarde de la meilleure solution + if (bCleanNonInformativeVariables) + dataGridManager.CopyInformativeDataGrid(&neighbourDataGrid, optimizedDataGrid); + else + dataGridManager.CopyDataGrid(&neighbourDataGrid, optimizedDataGrid); + + // Gestion de la meilleure solution + if (dataGridOptimizer != NULL) + dataGridOptimizer->HandleOptimizationStep(optimizedDataGrid, initialDataGrid, false); + } + // Sinon: on passe a un niveau de voisinnage plus fin + else + nIndex++; + + // Test de fin de tache + if (TaskProgression::IsInterruptionRequested()) + break; + + // Test si depassement de temps + if (IsOptimizationTimeElapsed()) + break; + + // Test cas d'une unique iteration + if (GetSlightOptimizationMode()) + break; + } + + ensure(fabs(dBestCost - dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid)) < dEpsilon); + return dBestCost; +} + +// CH IV Begin +double NEWKWDataGridVNSOptimizer::VNSDataGridPostOptimizeVarPart( + const KWDataGrid* initialDataGrid, KWDataGridMerger* neighbourDataGrid, double dNeighbourDataGridCost, + KWDataGrid* mergedDataGrid, KWDataGrid* partitionedReferencePostMergedDataGrid) const +{ + double dCost; + double dMergedCost; + double dFusionDeltaCost; + KWDataGridManager dataGridManager; + ALString sLabel; + boolean bDisplayResults = false; + + // On ne reverifie pas les precondition de la methode publique + require(initialDataGrid != NULL); + require(neighbourDataGrid != NULL); + require(neighbourDataGrid->GetDataGridCosts() == dataGridCosts); + require(fabs(dNeighbourDataGridCost - dataGridCosts->ComputeDataGridTotalCost(neighbourDataGrid)) < dEpsilon); + + // Initialisations + dataGridManager.SetSourceDataGrid(initialDataGrid); + dCost = dNeighbourDataGridCost; + dMergedCost = dNeighbourDataGridCost; + + // CH AB AF a voir avec Marc + // Cas ou l'on peut fusionner les parties de variable des cluster + // Probleme rencontre pour fusionner une grille non informative + // Pas de fusion ici mais question qui se pose : pourquoi dans OptimizeMerge on utilise systematiquement + // un CopyInformativeVariables et dans les methodes qui l'appellent cela est fait en fonction de + // bCleanNonInformativeVariables qui est a false + if (neighbourDataGrid->GetInformativeAttributeNumber() > 0 and optimizationParameters.GetVarPartPostMerge()) + { + // Tri des attributs + neighbourDataGrid->SortAttributeParts(); + + dataGridManager.SetSourceDataGrid(neighbourDataGrid); + // Creation d'une nouvelle grille avec nouvelle description des PV fusionnees + dFusionDeltaCost = + dataGridManager.ExportDataGridWithVarPartMergeOptimization(mergedDataGrid, dataGridCosts); + assert(not mergedDataGrid->GetVarPartsShared()); + + // Calcul et verification du cout + dMergedCost = dCost + dFusionDeltaCost; + + // Cas ou le cout de la grille avec PV voisines fusionnees est plus eleve que le cout avant + // fusion + if (dMergedCost > dCost * (1 + dEpsilon) and bDisplayResults) + { + sLabel = "PROBLEME : degradation du cout lors de la fusion des parties de variables " + "contigues"; + sLabel += DoubleToString(dMergedCost); + sLabel += "\t"; + sLabel += DoubleToString(dMergedCost - dCost); + AddWarning(sLabel); + cout << "PROBLEME : degradation du cout lors de la fusion des parties de variables " + "contigues : cout fusionne\t " + << dMergedCost << "\tcout\t" << dCost << "\n"; + cout << "Grille avant fusion\n"; + neighbourDataGrid->Write(cout); + cout << "Grille apres fusion\n"; + mergedDataGrid->Write(cout); + } + + // Post-optimisation de l'attribut VarPart uniquement dans le cas d'une optimisation approfondie + // (i.e. derniere granularite) + if (mergedDataGrid->GetInformativeAttributeNumber() > 1 and + optimizationParameters.GetVarPartPostOptimize() and not GetSlightOptimizationMode()) + { + CCVarPartDataGridPostOptimizer varPartDataGridPostOptimizer; + KWDataGrid mergedMergedDataGrid; + IntVector ivGroups; + ALString sInnerAttributeName; + double dVarPartFusionDeltaCost; + double dNewMergedCost; + double dMergedMergedCost; + int nGroupNumber; + boolean bImprovement; + + // Parametrage pour l'attribut VarPart + varPartDataGridPostOptimizer.SetPostOptimizationAttributeName( + mergedDataGrid->GetVarPartAttribute()->GetAttributeName()); + + if (bDisplayResults) + { + cout << "VNSOptimizeVarPartDataGrid : grille a post-optimiser" << endl; + mergedDataGrid->Write(cout); + cout << "Debut PostOptimisation VarPart" << endl; + cout << "VNSOptimizeVarPartDataGrid: grille initiale du DataGridOptimizer" << endl; + GetDataGridOptimizer()->GetInitialVarPartDataGrid()->Write(cout); + cout << "VNSOptimizeVarPartDataGrid: grille initiale utilisee pour l'export" << endl; + initialDataGrid->Write(cout); + cout << flush; + } + + // Boucle : on continue a post-optimiser tant qu'au moins un deplacement de VarPart + // permet d'ameliorer le critere + bImprovement = true; + int nImprovementNumber = 0; + while (bImprovement) + { + nImprovementNumber++; + + // Construction d'une grille de reference avec des clusters contenant une seule + // PV a partir des PV apres fusion Parametrage par la grille initiale de + // l'optimiseur + dataGridManager.SetSourceDataGrid(GetDataGridOptimizer()->GetInitialVarPartDataGrid()); + assert(mergedDataGrid->Check()); + dataGridManager.ExportDataGridWithSingletonVarParts( + mergedDataGrid, partitionedReferencePostMergedDataGrid, false); + ivGroups.SetSize( + partitionedReferencePostMergedDataGrid->GetVarPartAttribute()->GetPartNumber()); + assert(partitionedReferencePostMergedDataGrid->Check()); + assert(partitionedReferencePostMergedDataGrid->GetVarPartsShared()); + assert(partitionedReferencePostMergedDataGrid->GetInnerAttributes() == + mergedDataGrid->GetInnerAttributes()); + + // Affichage de la grille courante + if (bDisplayResults) + { + cout << "nImprovementNumber\t" << nImprovementNumber << endl; + cout << "VNSOptimizeVarPartDataGrid: grille post-fusionnee cout\t" + << dMergedCost << endl; + mergedDataGrid->Write(cout); + cout << "VNSOptimizeVarPartDataGrid : grille de reference" << endl; + partitionedReferencePostMergedDataGrid->Write(cout); + cout << flush; + } + + // Parametrage du profiling + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Post-optimization IV"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Improvement number", + IntToString(nImprovementNumber)); + + // Exploration des deplacements pour tous les attributs + bImprovement = varPartDataGridPostOptimizer.PostOptimizeLightVarPartDataGrid( + partitionedReferencePostMergedDataGrid, mergedDataGrid, &ivGroups); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Post-optimization IV"); + + if (bImprovement) + { + // Mise a jour de la grille pour l'optimisation de cet attribut + nGroupNumber = mergedDataGrid->GetVarPartAttribute()->GetPartNumber(); + dataGridManager.SetSourceDataGrid(partitionedReferencePostMergedDataGrid); + dataGridManager.UpdateVarPartDataGridFromVarPartGroups(mergedDataGrid, + &ivGroups, nGroupNumber); + + if (bDisplayResults) + { + dNewMergedCost = + dataGridCosts->ComputeDataGridTotalCost(mergedDataGrid); + cout << "VNSOptimizeVarPartDataGrid: grille mise a jour best " + "deplacement\tCout\t" + << dNewMergedCost << endl; + mergedDataGrid->Write(cout); + assert(mergedDataGrid->Check()); + } + + // Mise a jour de la grille fusionnee courante par la grille obtenue par + // fusion de la grille comportant les deplacements + dataGridManager.SetSourceDataGrid(mergedDataGrid); + dVarPartFusionDeltaCost = + dataGridManager.ExportDataGridWithVarPartMergeOptimization( + &mergedMergedDataGrid, dataGridCosts); + dMergedMergedCost = + dataGridCosts->ComputeDataGridTotalCost(&mergedMergedDataGrid); + if (bDisplayResults) + { + cout << "VNSOptimizeVarPartDataGrid: grille best deplacement " + "et fusionnee\tCout\t" + << dMergedMergedCost << endl; + mergedMergedDataGrid.Write(cout); + } + + // Cas ou la post-optimisation permet d'ameliorer le cout + if (dMergedMergedCost < dMergedCost) + { + dMergedCost = dMergedMergedCost; + + // Export de la grille avec les clusters post-optimises et les + // PV de la grille initiale. Grille sans fusion des PV voisines. + // C'est cette grille antecedent de la meilleure grille qui sera + // memorisee + neighbourDataGrid->DeleteAll(); + dataGridManager.SetSourceDataGrid(initialDataGrid); + dataGridManager.ExportDataGridWithReferenceVarPartClusters( + mergedDataGrid, neighbourDataGrid); + if (bDisplayResults) + { + cout << "VNSOptimizeVarPartDataGrid: grille best " + "deplacement et initiale" + << endl; + neighbourDataGrid->Write(cout); + } + dCost = dataGridCosts->ComputeDataGridTotalCost(neighbourDataGrid); + } + + // On remplace mergedDataGrid par la nouvelle grille integrant les + // deplacements de post-optimisation puis fusionnee 23/11/22 : pourquoi + // effectuer ce remplacement systematiquement et pas uniquement dans le + // cas ou la poste optimisation permet d'ameliorer le cout ? on est + // quand meme dans le cas if(bImprovement) donc il y a amelioration du + // fait de la post-optimisation + mergedDataGrid->DeleteAll(); + dataGridManager.SetSourceDataGrid(&mergedMergedDataGrid); + dataGridManager.ExportDataGrid(mergedDataGrid); + mergedDataGrid->SetVarPartsShared(false); + mergedMergedDataGrid.SetVarPartsShared(true); + mergedMergedDataGrid.DeleteAll(); + assert(mergedDataGrid->Check()); + } + // Nettoyage -> peut etre a supprimer pour mutualiser avec utilisation de cette + // grille pour HangleOptimizationStep + partitionedReferencePostMergedDataGrid->DeleteAll(); + ivGroups.SetSize(0); + } + if (bDisplayResults) + cout << "Fin PostOptimisation VarPart" << endl; + } + } + ensure(fabs(dCost - dataGridCosts->ComputeDataGridTotalCost(neighbourDataGrid)) < dEpsilon); + return dMergedCost; +} + +double NEWKWDataGridVNSOptimizer::VNSOptimizeVarPartDataGrid(const KWDataGrid* initialDataGrid, double dDecreaseFactor, + int nMinIndex, int nMaxIndex, + KWDataGrid* optimizedDataGrid, + double dOptimizedDataGridCost, + double& dBestMergedDataGridCost) const +{ + double dBestCost; + double dCost; + double dMergedCost; + double dBestMergedCost; + KWDataGrid mergedDataGrid; + KWDataGrid partitionedReferencePostMergedDataGrid; + KWDataGridManager dataGridManager; + KWDataGridMerger neighbourDataGrid; + int nIndex; + double dNeighbourhoodSize; + ALString sLabel; + + // On ne reverifie pas les precondition de la methode publique + require(dDecreaseFactor > 1); + require(0 <= nMinIndex); + require(nMinIndex <= nMaxIndex); + require(fabs(dOptimizedDataGridCost - dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid)) < dEpsilon); + + // CH IV Refactoring : DDDDD + // Test du remplacement de la methode actuelle, par son proto + boolean bDeprecatedVersion = false; + if (bDeprecatedVersion) + return DEPRECATED_VNSOptimizeVarPartDataGrid(initialDataGrid, dDecreaseFactor, nMinIndex, nMaxIndex, + optimizedDataGrid, dOptimizedDataGridCost, + dBestMergedDataGridCost); + + // Initialisations + dataGridManager.SetSourceDataGrid(initialDataGrid); + neighbourDataGrid.SetDataGridCosts(dataGridCosts); + dBestCost = dOptimizedDataGridCost; + dBestMergedCost = dBestCost; + + // On optimise tant qu'on ne depasse pas la taille max de voisinage + nVNSMaxLevel = nMaxIndex; + nIndex = nMinIndex; + while (nIndex <= nMaxIndex) + { + nVNSLevel = nIndex; + dNeighbourhoodSize = pow(1.0 / dDecreaseFactor, nIndex); + dVNSNeighbourhoodSize = dNeighbourhoodSize; + + // Generation d'une solution dans un voisinnage de la meilleure solution + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Generate neighbour solution"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Neighbourhood size", + DoubleToString(dNeighbourhoodSize)); + GenerateNeighbourSolution(initialDataGrid, optimizedDataGrid, dNeighbourhoodSize, &neighbourDataGrid); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Generate neighbour solution"); + + // Parametrage du profiling pour l'optimisation + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Optimize solution"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Index", IntToString(nIndex)); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("VNS neighbourhood size", + DoubleToString(dVNSNeighbourhoodSize)); + + // Optimisation de cette solution + dCost = OptimizeSolution(initialDataGrid, &neighbourDataGrid); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Optimize solution"); + + // Post-optimisation des parties de variables de la grille + // A terme, a deplacer dans OptimizeSolution + dMergedCost = dCost; + if (initialDataGrid->IsVarPartDataGrid()) + dMergedCost = + VNSDataGridPostOptimizeVarPart(initialDataGrid, &neighbourDataGrid, dCost, &mergedDataGrid, + &partitionedReferencePostMergedDataGrid); + + // Si amelioration: on la memorise + if (dMergedCost < dBestMergedCost - dEpsilon) + { + dBestCost = dCost; + dBestMergedCost = dMergedCost; + + // Sauvegarde de la meilleure solution + if (bCleanNonInformativeVariables) + { + // On sauvegarde l'antecedent de la meilleure grille post mergee + // On ne sauvegarde donc pas la meilleure grille post mergee car cela necessiterait de + // modifier la grille initiale qui devrait etre en coherence avec cette grille post + // mergee. La grille initiale est conservee pour une granularite donnee La grille de + // reference est necessaire pour HandleOptimizationStep + dataGridManager.CopyInformativeDataGrid(&neighbourDataGrid, optimizedDataGrid); + } + + else + { + dataGridManager.CopyDataGrid(&neighbourDataGrid, optimizedDataGrid); + } + + // Gestion de la meilleure solution + if (dataGridOptimizer != NULL) + { + if (mergedDataGrid.GetInformativeAttributeNumber() == 0) + dataGridOptimizer->HandleOptimizationStep(optimizedDataGrid, initialDataGrid, + false); + + else + { + dataGridManager.SetSourceDataGrid( + GetDataGridOptimizer()->GetInitialVarPartDataGrid()); + dataGridManager.ExportDataGridWithSingletonVarParts( + &mergedDataGrid, &partitionedReferencePostMergedDataGrid, true); + dataGridOptimizer->HandleOptimizationStep( + &mergedDataGrid, &partitionedReferencePostMergedDataGrid, false); + + // Nettoyage + partitionedReferencePostMergedDataGrid.DeleteAll(); + } + } + } + // Sinon: on passe a un niveau de voisinage plus fin + else + nIndex++; + + // Nettoyage + mergedDataGrid.DeleteAll(); + + // Test de fin de tache + if (TaskProgression::IsInterruptionRequested()) + break; + + // Test si depassement de temps + if (IsOptimizationTimeElapsed()) + break; + + // Test cas d'une unique iteration + if (GetSlightOptimizationMode()) + break; + } + + // Memorisation du meilleur cout parmi les grilles post-fusionnees + dBestMergedDataGridCost = dBestMergedCost; + ensure(fabs(dBestCost - dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid)) < dEpsilon); + return dBestCost; +} + +double NEWKWDataGridVNSOptimizer::DEPRECATED_VNSOptimizeVarPartDataGrid(const KWDataGrid* initialDataGrid, + double dDecreaseFactor, int nMinIndex, + int nMaxIndex, KWDataGrid* optimizedDataGrid, + double dOptimizedDataGridCost, + double& dBestMergedDataGridCost) const +{ + double dBestCost; + double dCost; + double dMergedCost; + double dBestMergedCost; + double dFusionDeltaCost; + KWDataGrid mergedDataGrid; + KWDataGrid partitionedReferencePostMergedDataGrid; + KWDataGridManager dataGridManager; + KWDataGridMerger neighbourDataGrid; + int nIndex; + double dNeighbourhoodSize; + ALString sLabel; + boolean bDisplayResults = false; + + // On ne reverifie pas les precondition de la methode publique + require(dDecreaseFactor > 1); + require(0 <= nMinIndex); + require(nMinIndex <= nMaxIndex); + require(fabs(dOptimizedDataGridCost - dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid)) < dEpsilon); + + // Initialisations + dataGridManager.SetSourceDataGrid(initialDataGrid); + neighbourDataGrid.SetDataGridCosts(dataGridCosts); + dBestCost = dOptimizedDataGridCost; + dBestMergedCost = dBestCost; + + // On optimise tant qu'on ne depasse pas la taille max de voisinage + nVNSMaxLevel = nMaxIndex; + nIndex = nMinIndex; + while (nIndex <= nMaxIndex) + { + nVNSLevel = nIndex; + dNeighbourhoodSize = pow(1.0 / dDecreaseFactor, nIndex); + dVNSNeighbourhoodSize = dNeighbourhoodSize; + + // Generation d'une solution dans un voisinnage de la meilleure solution + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Generate neighbour solution"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Neighbourhood size", + DoubleToString(dNeighbourhoodSize)); + GenerateNeighbourSolution(initialDataGrid, optimizedDataGrid, dNeighbourhoodSize, &neighbourDataGrid); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Generate neighbour solution"); + + // Parametrage du profiling pour l'optimisation + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Optimize solution"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("Index", IntToString(nIndex)); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString("VNS neighbourhood size", + DoubleToString(dVNSNeighbourhoodSize)); + + // Optimisation de cette solution + dCost = OptimizeSolution(initialDataGrid, &neighbourDataGrid); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Optimize solution"); + + // CH AB AF a voir avec Marc + // Cas ou l'on peut fusionner les parties de variable des cluster + // Probleme rencontre pour fusionner une grille non informative + // Pas de fusion ici mais question qui se pose : pourquoi dans OptimizeMerge on utilise systematiquement + // un CopyInformativeVariables et dans les methodes qui l'appellent cela est fait en fonction de + // bCleanNonInformativeVariables qui est a false + if (neighbourDataGrid.GetInformativeAttributeNumber() > 0 and + optimizationParameters.GetVarPartPostMerge()) + { + // Tri des attributs + neighbourDataGrid.SortAttributeParts(); + + dataGridManager.SetSourceDataGrid(&neighbourDataGrid); + // Creation d'une nouvelle grille avec nouvelle description des PV fusionnees + dFusionDeltaCost = + dataGridManager.ExportDataGridWithVarPartMergeOptimization(&mergedDataGrid, dataGridCosts); + assert(not mergedDataGrid.GetVarPartsShared()); + + // Calcul et verification du cout + dMergedCost = dCost + dFusionDeltaCost; + + // Cas ou le cout de la grille avec PV voisines fusionnees est plus eleve que le cout avant + // fusion + if (bDisplayResults and dMergedCost > dCost * (1 + dEpsilon)) + { + sLabel = "PROBLEME : degradation du cout lors de la fusion des parties de variables " + "contigues"; + sLabel += DoubleToString(dMergedCost); + sLabel += "\t"; + sLabel += DoubleToString(dMergedCost - dCost); + AddWarning(sLabel); + cout << "PROBLEME : degradation du cout lors de la fusion des parties de variables " + "contigues : cout fusionne\t " + << dMergedCost << "\tcout\t" << dCost << "\n"; + cout << "Grille avant fusion\n"; + neighbourDataGrid.Write(cout); + cout << "Grille apres fusion\n"; + mergedDataGrid.Write(cout); + } + + // Post-optimisation de l'attribut VarPart uniquement dans le cas d'une optimisation approfondie + // (i.e. derniere granularite) + if (mergedDataGrid.GetInformativeAttributeNumber() > 1 and + optimizationParameters.GetVarPartPostOptimize() and not GetSlightOptimizationMode()) + { + CCVarPartDataGridPostOptimizer varPartDataGridPostOptimizer; + KWDataGrid mergedMergedDataGrid; + IntVector ivGroups; + ALString sInnerAttributeName; + double dVarPartFusionDeltaCost; + double dNewMergedCost; + double dMergedMergedCost; + int nGroupNumber; + boolean bImprovement; + + // Parametrage pour l'attribut VarPart + varPartDataGridPostOptimizer.SetPostOptimizationAttributeName( + mergedDataGrid.GetVarPartAttribute()->GetAttributeName()); + + if (bDisplayResults) + { + cout << "VNSOptimizeVarPartDataGrid : grille a post-optimiser" << endl; + mergedDataGrid.Write(cout); + cout << "Debut PostOptimisation VarPart" << endl; + cout << "VNSOptimizeVarPartDataGrid: grille initiale du DataGridOptimizer" + << endl; + GetDataGridOptimizer()->GetInitialVarPartDataGrid()->Write(cout); + cout << "VNSOptimizeVarPartDataGrid: grille initiale utilisee pour l'export" + << endl; + initialDataGrid->Write(cout); + cout << flush; + } + + // Boucle : on continue a post-optimiser tant qu'au moins un deplacement de VarPart + // permet d'ameliorer le critere + bImprovement = true; + + int nImprovementNumber = 0; + while (bImprovement) + { + nImprovementNumber++; + + // Construction d'une grille de reference avec des clusters contenant une seule + // PV a partir des PV apres fusion Parametrage par la grille initiale de + // l'optimiseur + dataGridManager.SetSourceDataGrid( + GetDataGridOptimizer()->GetInitialVarPartDataGrid()); + assert(mergedDataGrid.Check()); + dataGridManager.ExportDataGridWithSingletonVarParts( + &mergedDataGrid, &partitionedReferencePostMergedDataGrid, false); + ivGroups.SetSize(partitionedReferencePostMergedDataGrid.GetVarPartAttribute() + ->GetPartNumber()); + assert(partitionedReferencePostMergedDataGrid.Check()); + assert(partitionedReferencePostMergedDataGrid.GetVarPartsShared()); + assert(partitionedReferencePostMergedDataGrid.GetInnerAttributes() == + mergedDataGrid.GetInnerAttributes()); + + // Affichage de la grille courante + if (bDisplayResults) + { + cout << "nImprovementNumber\t" << nImprovementNumber << endl; + cout << "VNSOptimizeVarPartDataGrid: grille post-fusionnee cout\t" + << dMergedCost << endl; + mergedDataGrid.Write(cout); + cout << "VNSOptimizeVarPartDataGrid : grille de reference" << endl; + partitionedReferencePostMergedDataGrid.Write(cout); + cout << flush; + } + + // Parametrage du profiling + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Post-optimization IV"); + NEWKWDataGridOptimizer::GetProfiler()->WriteKeyString( + "Improvement number", IntToString(nImprovementNumber)); + + // Exploration des deplacements pour tous les attributs + bImprovement = varPartDataGridPostOptimizer.PostOptimizeLightVarPartDataGrid( + &partitionedReferencePostMergedDataGrid, &mergedDataGrid, &ivGroups); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Post-optimization IV"); + + if (bImprovement) + { + // Mise a jour de la grille pour l'optimisation de cet attribut + nGroupNumber = mergedDataGrid.GetVarPartAttribute()->GetPartNumber(); + dataGridManager.SetSourceDataGrid( + &partitionedReferencePostMergedDataGrid); + dataGridManager.UpdateVarPartDataGridFromVarPartGroups( + &mergedDataGrid, &ivGroups, nGroupNumber); + + if (bDisplayResults) + { + dNewMergedCost = + dataGridCosts->ComputeDataGridTotalCost(&mergedDataGrid); + cout << "VNSOptimizeVarPartDataGrid: grille mise a jour best " + "deplacement\tCout\t" + << dNewMergedCost << endl; + mergedDataGrid.Write(cout); + assert(mergedDataGrid.Check()); + } + + // Mise a jour de la grille fusionnee courante par la grille obtenue par + // fusion de la grille comportant les deplacements + dataGridManager.SetSourceDataGrid(&mergedDataGrid); + dVarPartFusionDeltaCost = + dataGridManager.ExportDataGridWithVarPartMergeOptimization( + &mergedMergedDataGrid, dataGridCosts); + dMergedMergedCost = + dataGridCosts->ComputeDataGridTotalCost(&mergedMergedDataGrid); + if (bDisplayResults) + { + cout << "VNSOptimizeVarPartDataGrid: grille best deplacement " + "et fusionnee\tCout\t" + << dMergedMergedCost << endl; + mergedMergedDataGrid.Write(cout); + } + + // Cas ou la post-optimisation permet d'ameliorer le cout + if (dMergedMergedCost < dMergedCost) + { + dMergedCost = dMergedMergedCost; + + // Export de la grille avec les clusters post-optimises et les + // PV de la grille initiale. Grille sans fusion des PV voisines. + // C'est cette grille antecedent de la meilleure grille qui sera + // memorisee + neighbourDataGrid.DeleteAll(); + dataGridManager.SetSourceDataGrid(initialDataGrid); + dataGridManager.ExportDataGridWithReferenceVarPartClusters( + &mergedDataGrid, &neighbourDataGrid); + if (bDisplayResults) + { + cout << "VNSOptimizeVarPartDataGrid: grille best " + "deplacement et initiale" + << endl; + neighbourDataGrid.Write(cout); + } + dCost = + dataGridCosts->ComputeDataGridTotalCost(&neighbourDataGrid); + } + + // On remplace mergedDataGrid par la nouvelle grille integrant les + // deplacements de post-optimisation puis fusionnee 23/11/22 : pourquoi + // effectuer ce remplacement systematiquement et pas uniquement dans le + // cas ou la poste optimisation permet d'ameliorer le cout ? on est + // quand meme dans le cas if(bImprovement) donc il y a amelioration du + // fait de la post-optimisation + mergedDataGrid.DeleteAll(); + dataGridManager.SetSourceDataGrid(&mergedMergedDataGrid); + dataGridManager.ExportDataGrid(&mergedDataGrid); + mergedDataGrid.SetVarPartsShared(false); + mergedMergedDataGrid.SetVarPartsShared(true); + mergedMergedDataGrid.DeleteAll(); + assert(mergedDataGrid.Check()); + } + // Nettoyage -> peut etre a supprimer pour mutualiser avec utilisation de cette + // grille pour HangleOptimizationStep + partitionedReferencePostMergedDataGrid.DeleteAll(); + ivGroups.SetSize(0); + } + if (bDisplayResults) + cout << "Fin PostOptimisation VarPart" << endl; + } + } + // Cas ou l'on ne fusionne pas les parties de variable mitoyenne + else + dMergedCost = dCost; + + // Si amelioration: on la memorise + if (dMergedCost < dBestMergedCost - dEpsilon) + { + dBestCost = dCost; + dBestMergedCost = dMergedCost; + + // Sauvegarde de la meilleure solution + if (bCleanNonInformativeVariables) + { + // On sauvegarde l'antecedent de la meilleure grille post mergee + // On ne sauvegarde donc pas la meilleure grille post mergee car cela necessiterait de + // modifier la grille initiale qui devrait etre en coherence avec cette grille post + // mergee. La grille initiale est conservee pour une granularite donnee La grille de + // reference est necessaire pour HandleOptimizationStep + dataGridManager.CopyInformativeDataGrid(&neighbourDataGrid, optimizedDataGrid); + } + else + { + dataGridManager.CopyDataGrid(&neighbourDataGrid, optimizedDataGrid); + } + + // Gestion de la meilleure solution + if (dataGridOptimizer != NULL) + { + if (mergedDataGrid.GetInformativeAttributeNumber() == 0) + dataGridOptimizer->HandleOptimizationStep(optimizedDataGrid, initialDataGrid, + false); + else + { + dataGridManager.SetSourceDataGrid( + GetDataGridOptimizer()->GetInitialVarPartDataGrid()); + dataGridManager.ExportDataGridWithSingletonVarParts( + &mergedDataGrid, &partitionedReferencePostMergedDataGrid, true); + dataGridOptimizer->HandleOptimizationStep( + &mergedDataGrid, &partitionedReferencePostMergedDataGrid, false); + + // Nettoyage + partitionedReferencePostMergedDataGrid.DeleteAll(); + } + } + } + + // Sinon: on passe a un niveau de voisinage plus fin + else + nIndex++; + + // Nettoyage + mergedDataGrid.DeleteAll(); + + // Test de fin de tache + if (TaskProgression::IsInterruptionRequested()) + break; + + // Test si depassement de temps + if (IsOptimizationTimeElapsed()) + break; + + // Test cas d'une unique iteration + if (GetSlightOptimizationMode()) + break; + } + + // Memorisation du meilleur cout parmi les grilles post-fusionnees + dBestMergedDataGridCost = dBestMergedCost; + + ensure(fabs(dBestCost - dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid)) < dEpsilon); + return dBestCost; +} +// CH IV End + +double NEWKWDataGridVNSOptimizer::OptimizeSolution(const KWDataGrid* initialDataGrid, + KWDataGridMerger* dataGridMerger) const +{ + KWDataGridPostOptimizer dataGridPostOptimizer; + double dCost; + boolean bDisplay = false; + + require(initialDataGrid != NULL); + require(dataGridMerger != NULL); + + // Initialisations + dataGridPostOptimizer.SetDataGridCosts(dataGridCosts); + nVNSIteration++; + + // Affichage du cout initial + DisplayOptimizationDetails(dataGridMerger, false); + + // Cout initial si aucune optimisation + dCost = DBL_MAX; + if (not optimizationParameters.GetOptimize() and not optimizationParameters.GetPreOptimize() and + not optimizationParameters.GetPostOptimize()) + dCost = dataGridCosts->ComputeDataGridTotalCost(dataGridMerger); + + if (bDisplay) + cout << "Affichage de l'evolution des couts" << dCost << "\n"; + + // Pre-optimisation de la grille + if (optimizationParameters.GetPreOptimize() and + not TaskProgression::IsInterruptionRequested() + // CH RefontePrior2-P-Inside + and initialDataGrid->GetAttributeNumber() > 1) + // Fin CH RefontePrior2 + { + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Pre-optimization"); + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, dataGridMerger, false); + if (bDisplay) + cout << dCost << "\n"; + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Pre-optimization"); + } + + // Optimisation par fusion des groupes + if (optimizationParameters.GetOptimize() and not TaskProgression::IsInterruptionRequested()) + { + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Greedy merge optimization"); + dCost = dataGridMerger->Merge(); + if (bDisplay) + cout << dCost << "\n"; + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Greedy merge optimization"); + } + // Post-optimisation de la grille + if (optimizationParameters.GetPostOptimize() and + not TaskProgression::IsInterruptionRequested() + // CH RefontePrior2-P-Inside + and initialDataGrid->GetAttributeNumber() > 1) + // Fin CH RefontePrior2) + { + NEWKWDataGridOptimizer::GetProfiler()->BeginMethod("Post-optimization"); + // Cas d'une optimisation legere (pour les granularites intermediaires en co-clustering) + if (GetSlightOptimizationMode()) + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, dataGridMerger, false); + // Sinon + else + dCost = dataGridPostOptimizer.PostOptimizeDataGrid(initialDataGrid, dataGridMerger, true); + NEWKWDataGridOptimizer::GetProfiler()->EndMethod("Post-optimization"); + if (bDisplay) + cout << dCost << "\n\n"; + } + + // Recalcul du cout si la tache est interrompue, pour sortir avec un cout coherent + if (TaskProgression::IsInterruptionRequested()) + dCost = dataGridCosts->ComputeDataGridTotalCost(dataGridMerger); + + // Affichage du cout final + DisplayOptimizationDetails(dataGridMerger, true); + + // Retour du cout + ensure(fabs(dataGridCosts->ComputeDataGridTotalCost(dataGridMerger) - dCost) < dEpsilon); + return dCost; +} + +void NEWKWDataGridVNSOptimizer::GenerateNeighbourSolution(const KWDataGrid* initialDataGrid, + const KWDataGrid* optimizedDataGrid, double dNoiseRate, + KWDataGridMerger* neighbourDataGridMerger) const +{ + KWDataGridManager dataGridManager; + KWDataGrid mandatoryDataGrid; + int nMandatoryAttributeNumber; + int nMaxAttributeNumber; + int nMaxPartNumber; + int nMaxContinuousPartNumber; + int nMaxSymbolPartNumber; + int nRequestedContinuousPartNumber; + int nRequestedSymbolPartNumber; + int nAttributeNumber; + ALString sTmp; + // CH RefontePrior2 + int nGridSize; + // Fin CH RefontePrior2 + + require(initialDataGrid != NULL); + require(optimizedDataGrid != NULL); + require(neighbourDataGridMerger != NULL); + require(0 <= dNoiseRate and dNoiseRate <= 1); + + // Debut de tache + TaskProgression::BeginTask(); + TaskProgression::DisplayMainLabel(sTmp + "New initial solution (" + DoubleToString(dNoiseRate) + ")"); + + // Initialisation de la taille de la grille prise en compte + // Cas avec granularite + nGridSize = initialDataGrid->GetCellNumber(); + + // Calcul du nombre d'attributs a exporter + nMaxAttributeNumber = 1 + (int)(log(nGridSize * 1.0) / log(2.0)); + nAttributeNumber = (int)(dNoiseRate * nMaxAttributeNumber); + if (nAttributeNumber < 2) + nAttributeNumber = 2; + if (nAttributeNumber > initialDataGrid->GetAttributeNumber()) + nAttributeNumber = initialDataGrid->GetAttributeNumber(); + + // Calcul des nombres de parties a exporter + nMaxContinuousPartNumber = (int)(nGridSize / log(nGridSize + 1.0)); + nMaxSymbolPartNumber = (int)sqrt(nGridSize * 1.0); + nMaxPartNumber = (int)pow(nGridSize * 1.0, 1.0 / nAttributeNumber); + if (nMaxPartNumber > nGridSize) + nMaxPartNumber = nGridSize; + if (nMaxPartNumber < 2) + nMaxPartNumber = 2; + if (nMaxContinuousPartNumber > nMaxPartNumber) + nMaxContinuousPartNumber = nMaxPartNumber; + if (nMaxSymbolPartNumber > nMaxPartNumber) + nMaxSymbolPartNumber = nMaxPartNumber; + nRequestedContinuousPartNumber = 1 + (int)(dNoiseRate * nMaxContinuousPartNumber); + nRequestedSymbolPartNumber = 1 + (int)(dNoiseRate * nMaxSymbolPartNumber); + + // Parametrage avance temporaire, pour etude sur les multinomiales hierarchiques (Marc Boulle) + // Cf. classe d'etude KWHierarchicalMultinomialStudy + if (optimizationParameters.GetInternalParameter() == "LargeNeighborhoods") + { + nMaxContinuousPartNumber = (int)(nGridSize / 2); + nMaxSymbolPartNumber = (int)sqrt(nGridSize * 1.0); + nRequestedContinuousPartNumber = 1 + (int)(dNoiseRate * nMaxContinuousPartNumber); + nRequestedSymbolPartNumber = 1 + (int)(dNoiseRate * nMaxSymbolPartNumber); + } + + // Export d'un sous-ensemble d'attributs obligatoires (les attribut informatifs) en fonction du niveau de bruit + // Les attributs obligatoires sont les attributs de la grille optimisee que l'on va conserver + // Quand la grille initiale contient plus d'attributs que la grille optimisee (on ne peut pas toujours prendre + // tous les attributs pour des questions de complexite algorithmique), on n'en conserve que certains et on + // complete avec de nouveaux attributs absents de la grille optimisee Quand le NoiseRate est eleve (=1) alors + // aucun des attributs n'est conserve avec ce statut d'obligatoire + nMandatoryAttributeNumber = (int)ceil((1 - dNoiseRate) * optimizedDataGrid->GetAttributeNumber()); + dataGridManager.SetSourceDataGrid(optimizedDataGrid); + dataGridManager.ExportRandomAttributes(&mandatoryDataGrid, nMandatoryAttributeNumber); + + // Exports d'attributs supplementaires + // Pour un dNoiseRate de 1, tous les attributs sont ajoutes a concurrence de nMaxAttributeNumber + dataGridManager.SetSourceDataGrid(initialDataGrid); + neighbourDataGridMerger->DeleteAll(); + dataGridManager.AddRandomAttributes(neighbourDataGridMerger, &mandatoryDataGrid, nAttributeNumber); + + // Export des parties + dataGridManager.AddRandomParts(neighbourDataGridMerger, optimizedDataGrid, nRequestedContinuousPartNumber, + nRequestedSymbolPartNumber, 1.0); + TaskProgression::DisplayProgression(25); + // Export des cellules + dataGridManager.ExportCells(neighbourDataGridMerger); + + // Fin de tache + TaskProgression::EndTask(); +} + +void NEWKWDataGridVNSOptimizer::DisplayOptimizationHeaderLine() const +{ + if (optimizationParameters.GetDisplayDetails()) + { + // Lignes d'entete + cout << "Time\tIter\tNeigh. size\t"; + cout << "Initial\t\t\t\tFinal\t\t\t\t\n"; + cout << "\t\t\tAtt. number\tPart number\tCell number\tCost\t"; + cout << "Att. number\tPart number\tCell number\tCost\n"; + } +} + +void NEWKWDataGridVNSOptimizer::DisplayOptimizationDetails(const KWDataGrid* optimizedDataGrid, + boolean bOptimized) const +{ + double dVNSTime; + + // Calcul du temps ecoule + timerVNS.Stop(); + dVNSTime = timerVNS.GetElapsedTime(); + timerVNS.Start(); + + // Gestion de l'avancement avant chaque etape d'iteration + if (not bOptimized) + { + ALString sTmp; + int nTotalIterLevel; + int nProgressionIterLevel; + int nGranularityMax; + + if (optimizedDataGrid->GetGranularity() > 0) + nGranularityMax = (int)ceil(log(optimizedDataGrid->GetGridFrequency() * 1.0) / log(2.0)); + else + nGranularityMax = 0; + + // Message + + // CH RefontePrior2-G + if (optimizedDataGrid->GetGranularity() == 0 or optimizedDataGrid->GetGranularity() == nGranularityMax) + TaskProgression::DisplayLabel(sTmp + " VNS " + IntToString(nVNSIteration) + " " + + IntToString(nVNSLevel) + "/" + IntToString(nVNSMaxLevel) + " (" + + DoubleToString((int)(10000 * dVNSNeighbourhoodSize) / 10000.0) + + ")"); + else + TaskProgression::DisplayLabel( + sTmp + "Granularity " + IntToString(optimizedDataGrid->GetGranularity()) + "/" + + IntToString(nGranularityMax) + " VNS " + IntToString(nVNSIteration) + " " + + IntToString(nVNSLevel) + "/" + IntToString(nVNSMaxLevel) + " (" + + DoubleToString((int)(10000 * dVNSNeighbourhoodSize) / 10000.0) + ")"); + + // Niveau d'avancement + if (optimizationParameters.GetOptimizationTime() > 0) + { + if (dVNSTime > optimizationParameters.GetOptimizationTime()) + TaskProgression::DisplayProgression(100); + else + TaskProgression::DisplayProgression( + (int)(dVNSTime * 100 / optimizationParameters.GetOptimizationTime())); + } + // cas sans limite de temps + else + { + nTotalIterLevel = (int)pow(2.0, 2 + optimizationParameters.GetOptimizationLevel()); + nProgressionIterLevel = nVNSIteration; + while (nTotalIterLevel < nProgressionIterLevel) + nTotalIterLevel = nTotalIterLevel + (1 + nTotalIterLevel) / 2; + TaskProgression::DisplayProgression((int)(nProgressionIterLevel * 100.0 / nTotalIterLevel)); + } + } + + // Affichage des details d'optimisation + if (optimizationParameters.GetDisplayDetails()) + { + // Affichage de l'iteration + if (not bOptimized) + cout << dVNSTime << "\t" << nVNSIteration << "\t" << dVNSNeighbourhoodSize << "\t"; + + // Affichage des caracteristiques de la grille terminale + cout << optimizedDataGrid->GetAttributeNumber() << "\t" << optimizedDataGrid->GetTotalPartNumber() + << "\t" << optimizedDataGrid->GetCellNumber() << "\t" + << dataGridCosts->ComputeDataGridTotalCost(optimizedDataGrid); + if (not bOptimized) + cout << "\t"; + else + cout << "\n"; + cout << flush; + } +} + +boolean NEWKWDataGridVNSOptimizer::IsOptimizationTimeElapsed() const +{ + double dVNSTime; + + require(timerVNS.IsStarted()); + + if (optimizationParameters.GetOptimizationTime() == 0) + return false; + else + { + // Calcul du temps ecoule + timerVNS.Stop(); + dVNSTime = timerVNS.GetElapsedTime(); + timerVNS.Start(); + + // Test de depassement + return dVNSTime >= optimizationParameters.GetOptimizationTime(); + } +} diff --git a/src/Learning/KWDataPreparation/NEWKWDataGridOptimizer.h b/src/Learning/KWDataPreparation/NEWKWDataGridOptimizer.h new file mode 100644 index 000000000..0f5b421fa --- /dev/null +++ b/src/Learning/KWDataPreparation/NEWKWDataGridOptimizer.h @@ -0,0 +1,336 @@ +// Copyright (c) 2023 Orange. All rights reserved. +// This software is distributed under the BSD 3-Clause-clear License, the text of which is available +// at https://spdx.org/licenses/BSD-3-Clause-Clear.html or see the "LICENSE" file for more details. + +#pragma once + +class NEWKWDataGridOptimizer; +class NEWKWDataGridVNSOptimizer; +class CCCoclusteringOptimizer; + +#include "KWClassStats.h" +#include "KWDataGrid.h" +#include "KWDataGridCosts.h" +#include "KWDataGridMerger.h" +#include "KWDataGridManager.h" +#include "KWDataGridPostOptimizer.h" +#include "KWDataGridOptimizerParameters.h" +#include "SortedList.h" +#include "Profiler.h" +#include "Timer.h" + +////////////////////////////////////////////////////////////////////////////////// +// Classe NEWKWDataGridOptimizer +// Optimisation d'une grille de donnees parametree par une structure de cout. +class NEWKWDataGridOptimizer : public Object +{ +public: + // Constructeur + NEWKWDataGridOptimizer(); + ~NEWKWDataGridOptimizer(); + + // Reinitialisation + virtual void Reset(); + + // Parametrage de la structure des couts de la grille de donnees + // Memoire: les specifications sont referencees et destinees a etre partagees par plusieurs algorithmes + void SetDataGridCosts(const KWDataGridCosts* kwdgcCosts); + const KWDataGridCosts* GetDataGridCosts() const; + + // Parametres d'optimisation + // Memoire: l'objet rendu appartient a l'appele + KWDataGridOptimizerParameters* GetParameters(); + + // Parametrage (facultatif) par des statistiques sur le probleme d'apprentissage + // Permet l'utilisation des statistiques univariees pour optimiser les grilles de donnees + // Memoire: les specifications sont referencees et destinee a etre partagees + void SetClassStats(KWClassStats* stats); + KWClassStats* GetClassStats() const; + + // CH IV Begin + // Parametrage (facultatif) par une grille initiale, dans le cas du coclustering instances * variables, + // Permet l'utilisation de cette grille pour la creation de la grille avec parties de variables fusionnees + // Memoire: les specifications sont referencees et destinees a etre partagees + void SetInitialVarPartDataGrid(KWDataGrid* refDataGrid); + KWDataGrid* GetInitialVarPartDataGrid() const; + // CH IV End + + // Optimisation d'un grille pour une structure de cout donnees + // En sortie, on trouve une nouvelle grille optimisee compatible avec la grille initiale, + // ne conservant que les attributs non reduits a une seule partie + // Les intervalles (resp. groupes) de la grille optimisee sont tries par valeur (resp. effectifs decroissants) + // Integre un parcours des granularites + // Retourne le cout de codage MODL de la grille post-optimisee + // Integre un parcours des granularites + // Dans le cas d'une grille VarPart avec des parties de variable, le cout retourne est celui de la grille + // antecedente de la meilleure grille post-fusionnee (fusion des parties de variable consecutives dans un + // cluster de parties de variables) + double OptimizeDataGrid(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid) const; + + // Simplification d'une grille selon le parametre MaxPartNumber des parametres d'optimisation + // La grille en parametre est en simplifiee si necessaire, en fusionnant iterativement les partie + // tant que la contrainte de nombre max de partie n'est pas respectee + // Ce parametre n'est pris en compte qu'en etape de post-traitement d'un grille deja optimisee + // . cela permet de faire en sorte que les grilles simplifiee sont toutes des sous-grille de la grille + // optimale, + // ce qui simplifie l'interpretabilite + // . cela simplifie l'implementation, qui n'est ici effectuee qu'en post-traitement + // Retourne le cout de codage MODL de la grille simplifiee + double SimplifyDataGrid(KWDataGrid* optimizedDataGrid) const; + + ////////////////////////////////////////////////////////////////// + // Parametrage avance + + // Methode appelee lors de l'optimisation a la fin de chaque etape d'optimisation + // Permet par exemple de passer en mode anytime, en memorisant chaque solution intermediaire + // Par defaut: redirige vers la methode HandleOptimizationStep attributeSubsetStats s'il est parametre + virtual void HandleOptimizationStep(const KWDataGrid* optimizedDataGrid, + const KWDataGrid* initialGranularizedDataGrid, boolean bIsLastSaving) const; + + // Parametrage du contexte de gestion de la partie anytime de l'optimisation + // Permet de rediriger la methode HandleOptimizationStep vers celle du attributeSubsetStats + void SetAttributeSubsetStats(const KWAttributeSubsetStats* attributeSubsetStats); + const KWAttributeSubsetStats* GetAttributeSubsetStats(); + + ////////////////////////////////////////////////////////////////// + // Gestion d'un profiler dedie a l'optimisation des grilles + // Ce profiler doit etre demarre depuis le point d'entree de l'optimisation, + // et utilise par les methodes d'optimisation a profiler + + // Acces au profiler global permettant d'enregistrer toute une session de profiling + // de l'optimisation d'une grille de coclustering + static Profiler* GetProfiler(); + + ////////////////////////////////////////////////////////////////////////////////////////////// + ///// Implementation +protected: + // CH IV Refactoring: a supprimer + friend class KWDataGridOptimizer; + + // Optimisation d'une grille pour la granularite courante + // Ajout arguments bIsLastGranularity et dTotalComputeTime + double OptimizeGranularizedDataGrid(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid, + boolean bIsLastGranularity, double& dTotalComputeTime) const; + + // Post-optimisation de la granularite de la grille optimisee afin de lui attribuer la plus petite granularite + // avec laquelle la partition est compatible + void PostOptimizeGranularity(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid, + ObjectDictionary& odQuantileBuilders, int nLastExploredGranularity) const; + + // Initialisation avec une grille terminale + double InitializeWithTerminalDataGrid(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid) const; + + // Cette methode calcule pour chaque attribut de la grille initialDataGrid + // le partitionnement univarie optimal associe au partitionnement obtenu par projection univariee de la grille + // La granularite du partitionnement est celle de la grille initiale + double OptimizeWithBestUnivariatePartitionForCurrentGranularity(const KWDataGrid* initialDataGrid, + KWDataGrid* optimizedDataGrid) const; + + // Recherche d'une amelioration par croisement des partition univariees + // En mode hors granularite, la methode utilise directement les partitions univariees stockees dans KWClassStats + // En mode granularite, les partitions univariees optimales sont recalculees pour la granularite de la grille + // initiale et sont utilisees pour construire une grille produit + // bIsLastGranularity : Pour les granularites intermediaires, la post-optimisation n'est pas "deep" + double OptimizeWithMultipleUnivariatePartitions(const KWDataGrid* initialDataGrid, + KWDataGrid* optimizedDataGrid) const; + + // Optimisation greedy (DataGridMerger) + double GreedyOptimize(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid) const; + + // Optimisation en multi-start + double MultiStartOptimize(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid) const; + + // Optimisation avec VNS + // Traitement plus leger pour les granularites intermediaires en non supervise + double VNSOptimize(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid, + boolean bIsLastGranularity) const; + + // Affichage des caracteristiques d'optimisation (selon le niveau d'affichage demande) + void DisplayOptimizationHeaderLine() const; + void DisplayOptimizationDetails(const KWDataGrid* optimizedDataGrid, boolean bOptimized) const; + + // Parametrage de la structure des couts + const KWDataGridCosts* dataGridCosts; + + // Parametres d'optimisation + // Ajout mutable pour pouvoir modifier optimizationTime + mutable KWDataGridOptimizerParameters optimizationParameters; + + // Attribut de statistiques + KWClassStats* classStats; + + // CH IV Begin + // Grille de reference + KWDataGrid* initialVarPartDataGrid; + // CH IV End + + // Nettoyage des attribut non informatifs + boolean bCleanNonInformativeVariables; + + // Epsilon d'optimisation + double dEpsilon; + + // Contexte de gestion de la partie anytime de l'optimisation + const KWAttributeSubsetStats* attributeSubsetStatsHandler; + + // Profiler + static Profiler profiler; +}; + +////////////////////////////////////////////////////////////////////////////////// +// Classe NEWKWDataGridVNSOptimizer +// Optimisation d'une grille de donnees parametree par une structure de cout. +class NEWKWDataGridVNSOptimizer : public Object +{ +public: + // Constructeur + NEWKWDataGridVNSOptimizer(); + ~NEWKWDataGridVNSOptimizer(); + + // Parametrage de la structure des couts de la grille de donnees + // Memoire: les specifications sont referencees et destinees a etre partagees par plusieurs algorithmes + void SetDataGridCosts(const KWDataGridCosts* kwdgcCosts); + const KWDataGridCosts* GetDataGridCosts() const; + + // Parametres d'optimisation + // Memoire: l'objet rendu appartient a l'appele + KWDataGridOptimizerParameters* GetParameters(); + + // Mode d'optimisation legere utilise en coclustering pour les granularites intermediaires pour lesquelles + // il importe avant tout d'avoir une solution mais qu'il n'est pas necessaire de raffiner + void SetSlightOptimizationMode(boolean bValue); + const boolean GetSlightOptimizationMode() const; + + // Optimisation d'un grille pour une structure de cout donnees + // La grille initiale en entree contient la description la plus fine possible des + // partitions de chaque attribut. + // En entree, la grille optimisee constitue une solution de depart, compatible avec la grille initiale. + // En sortie, la grille optimisee est amelioree par la methode la meta-heuristique VNS + // Retourne le cout de codage MODL de la grille post-optimisee + double OptimizeDataGrid(const KWDataGrid* initialDataGrid, KWDataGrid* optimizedDataGrid) const; + + ////////////////////////////////////////////////////////////////// + // Parametrage avance + + // Parametrage (facultatif) de l'optimiseur maitre, pour avoir avoir acces a la methode HandleOptimizationStep + // appelee a chaque etape produisant une solution intermediaire lors de l'optimisation + void SetDataGridOptimizer(const NEWKWDataGridOptimizer* optimizer); + const NEWKWDataGridOptimizer* GetDataGridOptimizer() const; + + ////////////////////////////////////////////////////////////////////////////////////////////// + ///// Implementation +protected: + // Pilotage de la meta heuristique globale. + // Compromis entre exploration globale (rapartir de solution random) et exploitation locale + // d'un voisinnage autout de la meilleure solution + // OptimisationVNS(Level): + // . NeibourhoodSize = 1 pour une taille de voisinage initiale maximale (1 pour solution random) + // . repeter de niveau 1 a Level + // - tant que amelioration, generer solution dans voisinnage et optimisation + // - diviser la taille de voisinnage par un facteur donne + // Optimisation iterative(MaxLevel) + // . bestSolution = currentSolution + // . repeter de niveau 1 a MaxLevel + // - exploration globale de meme niveau que la meilleure solution courante: + // appeler OptimisationVNS(Level) en partant d'une nouvelle solution aleatoire (exploration globale) + // - on retient le meilleur de la solution courante et de la nouvelle solution exploree globalement (au + // meme niveau) + // - affinage de l'exploration locale au niveau superieur + double IterativeVNSOptimizeDataGrid(const KWDataGrid* initialDataGrid, int nMaxLevel, + KWDataGrid* optimizedDataGrid) const; + + // Pilotage de la meta heuristique VNS, avec des voisinnages successifs de taille decroissante + // selon un facteur geometrique (de DecreaseFactor^MinIndex a Factor^MaxIndex) + double VNSOptimizeDataGrid(const KWDataGrid* initialDataGrid, double dDecreaseFactor, int nMinIndex, + int nMaxIndex, KWDataGrid* optimizedDataGrid, double dOptimizedDataGridCost) const; + + // CH IV Refactoring + // Methode de post-optimisation d'un grille optimisee en redecoupant ses parties de variables + // Methode temporaire permettant de reutiliser le code de la methode principale VNSOptimizeDataGrid + // en isolant cette partie de post-optimisation specifique VarPart, et de supprimer l'ancienne + // methode VNSOptimizeVarPartDataGrid + // Parametre (a faire evoluer si necessaire): + // - initialDataGrid: grille initiale + // - neighbourDataGrid: grille courante en cours d'optimisation + // - dNeighbourDataGridCost: cout de la grille courante + // - mergedDataGrid: grille optimisee si amelioration + // En sortie, on rend la nouvelle version de la grille optimisee (ou courante) integrant la post-ptimisation et on renvoie son cout + // + // A terme, il faudra isoler ce service de post-optimisation pour le deplacer en quatrime sous-methode + // en fin de la methode OptimizeSolution + // - Pre-optimization + // - Greedy merge optimization + // - Post-optimization + // - Post-optimization IV + double VNSDataGridPostOptimizeVarPart(const KWDataGrid* initialDataGrid, KWDataGridMerger* neighbourDataGrid, + double dNeighbourDataGridCost, KWDataGrid* mergedDataGrid, + KWDataGrid* partitionedReferencePostMergedDataGrid) const; + + // CH IV Begin + // Pilotage de la meta heuristique VNS, avec des voisinages successifs de taille décroissante + // selon un facteur geometrique (de DecreaseFactor^MinIndex a Factor^MaxIndex) + // Les grilles generiques optimales sont post-fusionnees et les voisinages sont ceux des grilles antecedentes + // des grilles de meilleur cout apres post-fusion En sortie : optimizedDataGrid contient la grille antecedent + // avant post-fusion de la meilleure grille post-fusionne Le cout renvoye est le cout de cette optimizedDataGrid + // dBestMergedDataGridCost contient le cout de la meilleure grille apres post-fusion (meilleur cout) + // CH IV Refactoring: proto en vue de fusionner la methode avec VNSOptimizeDataGrid + double VNSOptimizeVarPartDataGrid(const KWDataGrid* initialDataGrid, double dDecreaseFactor, int nMinIndex, + int nMaxIndex, KWDataGrid* optimizedDataGrid, double dOptimizedDataGridCost, + double& dBestMergedDataGridCost) const; + // CH IV Refactoring: ancienne version avant decoupage en VNSOptimizeVarPartDataGrid et VNSDataGridPostOptimizeVarPart + double DEPRECATED_VNSOptimizeVarPartDataGrid(const KWDataGrid* initialDataGrid, double dDecreaseFactor, + int nMinIndex, int nMaxIndex, KWDataGrid* optimizedDataGrid, + double dOptimizedDataGridCost, + double& dBestMergedDataGridCost) const; + // CH IV End + + // Optimisation d'une solution + // Optimisation et post-optimisation activee selon les parametres d'optimisations + double OptimizeSolution(const KWDataGrid* initialDataGrid, KWDataGridMerger* dataGridMerger) const; + + // Creation d'une solution voisine d'une solution optimisee + // On passe en parametre un pourcentage de perturbation compris + // entre 0 (pas de perturbation) et 1 (solution aleatoire) + void GenerateNeighbourSolution(const KWDataGrid* initialDataGrid, const KWDataGrid* optimizedDataGrid, + double dNoiseRate, KWDataGridMerger* neighbourDataGridMerger) const; + + // Affichage des caracteristiques d'optimisation (selon le niveau d'affichage demande) + void DisplayOptimizationHeaderLine() const; + void DisplayOptimizationDetails(const KWDataGrid* optimizedDataGrid, boolean bOptimized) const; + + // Test si le temps d'optimisation est depasse (si le parametre correspondant est actif (non null)) + boolean IsOptimizationTimeElapsed() const; + + // Parametrage de la structure des couts + const KWDataGridCosts* dataGridCosts; + + // Parametres d'optimisation + KWDataGridOptimizerParameters optimizationParameters; + + // Execution d'une seule iteration VNS dans le cas d'une granularite intermediaire en co-clustering + boolean bSlightOptimizationMode; + + // Nettoyage des attribut non informatifs + boolean bCleanNonInformativeVariables; + + // Informations d'indexage des solutions evaluees durant l'optimisation + mutable Timer timerVNS; + mutable int nVNSIteration; + mutable int nVNSLevel; + mutable int nVNSMaxLevel; + mutable double dVNSNeighbourhoodSize; + + // Optimiseur principal pour declencher la methode de gestion des ameliorations + const NEWKWDataGridOptimizer* dataGridOptimizer; + + // Epsilon d'optimisation + double dEpsilon; +}; + +//////////////////////// +// Methodes en inline + +inline Profiler* NEWKWDataGridOptimizer::GetProfiler() +{ + return &profiler; +} diff --git a/src/Learning/MODL/MODL.cpp b/src/Learning/MODL/MODL.cpp index c379e44c4..2183fea95 100644 --- a/src/Learning/MODL/MODL.cpp +++ b/src/Learning/MODL/MODL.cpp @@ -35,6 +35,7 @@ int main(int argc, char** argv) // Choix du repertoire de lancement pour le debugage sous Windows (a commenter apres fin du debug) // SetWindowsDebugDir("Standard", "IrisLight"); + SetWindowsDebugDir("Standard", "Iris2D"); // Parametrage des logs memoires depuis les variables d'environnement, pris en compte dans KWLearningProject // KhiopsMemStatsLogFileName, KhiopsMemStatsLogFrequency, KhiopsMemStatsLogToCollect diff --git a/src/Learning/MODL_Coclustering/CCCoclusteringBuilder.cpp b/src/Learning/MODL_Coclustering/CCCoclusteringBuilder.cpp index cc8d649b4..346a2c036 100644 --- a/src/Learning/MODL_Coclustering/CCCoclusteringBuilder.cpp +++ b/src/Learning/MODL_Coclustering/CCCoclusteringBuilder.cpp @@ -352,7 +352,7 @@ boolean CCCoclusteringBuilder::ComputeCoclustering() boolean bProfileOptimisation = false; KWTupleTable tupleTable; KWTupleTable tupleFrequencyTable; - CCCoclusteringOptimizer dataGridOptimizer; + KWDataGridOptimizer dataGridOptimizer; KWDataGrid optimizedDataGrid; KWDataGridManager dataGridManager; ALString sProfileFileName; @@ -551,7 +551,7 @@ boolean CCCoclusteringBuilder::ComputeCoclustering() void CCCoclusteringBuilder::OptimizeDataGrid(const KWDataGrid* inputInitialDataGrid, KWDataGrid* optimizedDataGrid) { boolean bDisplayResults = false; - CCCoclusteringOptimizer dataGridOptimizer; + KWDataGridOptimizer dataGridOptimizer; require(inputInitialDataGrid != NULL); require(coclusteringDataGridCosts == NULL); @@ -578,7 +578,7 @@ void CCCoclusteringBuilder::OptimizeDataGrid(const KWDataGrid* inputInitialDataG } void CCCoclusteringBuilder::InitializeDataGridOptimizer(const KWDataGrid* inputInitialDataGrid, - CCCoclusteringOptimizer* dataGridOptimizer) + KWDataGridOptimizer* dataGridOptimizer) { require(inputInitialDataGrid != NULL); require(dataGridOptimizer != NULL); @@ -593,7 +593,7 @@ void CCCoclusteringBuilder::InitializeDataGridOptimizer(const KWDataGrid* inputI dataGridOptimizer->SetDataGridCosts(coclusteringDataGridCosts); // Parametrage pour l'optimisation anytime: avoir acces aux ameliorations a chaque etape de l'optimisation - dataGridOptimizer->SetCoclusteringBuilder(this); + dataGridOptimizer->SetAttributeSubsetStats(this); // Recopie du parametrage d'optimisation des grilles dataGridOptimizer->GetParameters()->CopyFrom(GetPreprocessingSpec()->GetDataGridOptimizerParameters()); @@ -614,7 +614,7 @@ void CCCoclusteringBuilder::OptimizeVarPartDataGrid(const KWDataGrid* inputIniti { boolean bDisplayResults = false; boolean bDisplayPartitionLevel = false; - CCCoclusteringOptimizer dataGridOptimizer; + KWDataGridOptimizer dataGridOptimizer; ObjectDictionary odInnerAttributesQuantileBuilders; KWDataGrid* nullDataGrid; KWDataGrid* partitionedDataGrid; @@ -4334,39 +4334,3 @@ void CCCoclusteringBuilder::SortAttributePartsAndValues(CCHierarchicalDataGrid* // Fin CH IV } } - -////////////////////////////////////////////////////////////////////////////////// -// Classe CCCoclusteringOptimizer - -CCCoclusteringOptimizer::CCCoclusteringOptimizer() -{ - coclusteringBuilder = NULL; -} - -CCCoclusteringOptimizer::~CCCoclusteringOptimizer() {} - -void CCCoclusteringOptimizer::Reset() -{ - KWDataGridOptimizer::Reset(); - coclusteringBuilder = NULL; -} - -void CCCoclusteringOptimizer::SetCoclusteringBuilder(const CCCoclusteringBuilder* builder) -{ - coclusteringBuilder = builder; -} - -const CCCoclusteringBuilder* CCCoclusteringOptimizer::GetCoclusteringBuilder() -{ - return coclusteringBuilder; -} - -void CCCoclusteringOptimizer::HandleOptimizationStep(const KWDataGrid* optimizedDataGrid, - const KWDataGrid* initialGranularizedDataGrid, - boolean bIsLastSaving) const -{ - // Integration de la granularite - if (coclusteringBuilder != NULL) - coclusteringBuilder->HandleOptimizationStep(optimizedDataGrid, initialGranularizedDataGrid, - bIsLastSaving); -} diff --git a/src/Learning/MODL_Coclustering/CCCoclusteringBuilder.h b/src/Learning/MODL_Coclustering/CCCoclusteringBuilder.h index 2ad4c8170..e85b19038 100644 --- a/src/Learning/MODL_Coclustering/CCCoclusteringBuilder.h +++ b/src/Learning/MODL_Coclustering/CCCoclusteringBuilder.h @@ -130,7 +130,7 @@ class CCCoclusteringBuilder : public KWAttributeSubsetStats // Supression du dernier fichier temporaire sauvegarde void RemoveLastSavedReportFile() const; - // Methode appelee lors de l'optimisation a chaque etape d'optimisation + // Methode redefinie, appelee lors de l'optimisation a chaque etape d'optimisation // A chaque amelioration, un nouveau fichier de sauvegarde est cree avec un index croissant, // le fichier precedent etant detruit. // Un message utilisateur est egalement emis. @@ -138,7 +138,7 @@ class CCCoclusteringBuilder : public KWAttributeSubsetStats // bIsLastSaving : si true, la sauvegarde est effectue meme s'il n'y a pas amelioration // Permet de recalculer la hierarchie du coclustering apres l'atteinte de la granularite maximale void HandleOptimizationStep(const KWDataGrid* optimizedDataGrid, const KWDataGrid* initialGranularizedDataGrid, - boolean bIsLastSaving) const; + boolean bIsLastSaving) const override; // Libelles utilisateur: nom du module de l'application (GetLearningModuleName()) const ALString GetClassLabel() const override; @@ -151,7 +151,8 @@ class CCCoclusteringBuilder : public KWAttributeSubsetStats // Initialisation d'un optimiseur de grille dedie coclustering void InitializeDataGridOptimizer(const KWDataGrid* inputInitialDataGrid, - CCCoclusteringOptimizer* dataGridOptimizer); + KWDataGridOptimizer* dataGridOptimizer); + //DDDD CCCoclusteringOptimizer* dataGridOptimizer); // CH IV Begin // Methode d'optimisation d'une grille dediee au cas instances x variables @@ -337,6 +338,7 @@ class CCCoclusteringBuilder : public KWAttributeSubsetStats boolean bExportAsKhc; }; +/*DDDD ////////////////////////////////////////////////////////////////////////////////// // Classe CCCoclusteringOptimizer // Specialisation de l'optimisation des grille pour passer en mode anytime @@ -364,3 +366,4 @@ class CCCoclusteringOptimizer : public KWDataGridOptimizer protected: const CCCoclusteringBuilder* coclusteringBuilder; }; +*/ \ No newline at end of file diff --git a/src/Learning/MODL_Coclustering/MODL_Coclustering.cpp b/src/Learning/MODL_Coclustering/MODL_Coclustering.cpp index 1fd1b50ab..e56f12d45 100644 --- a/src/Learning/MODL_Coclustering/MODL_Coclustering.cpp +++ b/src/Learning/MODL_Coclustering/MODL_Coclustering.cpp @@ -36,7 +36,7 @@ int main(int argc, char** argv) // Choix du repertoire de lancement pour le debugage sous Windows (a commenter apres fin du debug) //SetWindowsDebugDir("y_CoclusteringIV_Standard", "IrisLight"); //SetWindowsDebugDir("y_CoclusteringIV_Standard", "Iris"); - //SetWindowsDebugDir("Standard", "Iris"); + SetWindowsDebugDir("Standard", "Iris"); // Point d'arret sur l'allocation d'un bloc memoire // MemSetAllocIndexExit(77);