I found myself studying SOMs, mainly through Essentials of the Self-Organizing Map, 2013 by Teuvo Kohonen.
The main obstacle I faced was a clear way to tune the hyperparameters of the SOM. This guide is a summary of the main points I found useful to understand SOMs and how to tune them.
I will assume you have a basic understanding of SOMs. If you don't, I recommend reading the paper mentioned above.
Quantization error is the main metrics to be looked at when tuning the hyperparameters. It represents the average distance between each input vector and its corresponding winning neuron. If the quantization error is 0, then the SOM perfectly represents the input data.
Formula: [ QE = \frac{1}{N} \sum_{i=1}^{N} ||x_i - w_{c,i}|| ]
where:
-
is the number of observations -
is the -th sample -
is the weight vector of the winner neuron for the -th sample -
is the quantization error -
is the Euclidean distance
"The Quantization Error [...], declines as the map becomes larger. [...] Thus, the Quantization Error cannot be used to compare maps of different sizes." (Pölzlbauer, 2005)
Based on this principle, we will tune the hyperparameters based on maps of equal size.
Last but not least, the topographic error is calculated as the fraction of samples whose best and second-best matching units are not neighbors on the map. Formula retrived from [Forest, 2020]1:
[ \text{TE} = \frac{1}{N} \sum_{i=1}^{N} \mathbf{1}_{\delta(b1_i, b2_i) > 1} ]
where:
-
is the number of samples -
is the indicator function that is 1 if the first and second best matching units are not adjacent, 0 otherwise
Even though we will not use the topographic error to tune the hyperparameters, it is a good metric to evaluate the quality of the map. And, as we will do, we will always check its value after the training.
Some good heuristic to determine the number of neurons is, based on Vesanto, 2005, the following: $$ n_{neurons} = 5 \times \sqrt{N} $$ where:
-
is the number of observations
There are two main hyperparameters to be tuned:
- The learning rate
- The neighborhood function
Using the class HyperparameterTuning
we can perform a grid search to find the best hyperparameters, based on the quantization error metric. The class uses the minisom
package.
Example usage:
from src.utils.Hyperparameter import HyperparameterTuning
tuning = HyperparameterTuning() # If n_neurons is not passed, it will be calculated based on the heuristic mentioned above
best_params = tuning.run()
print(best_params)
Moreover, we can visualize the map using the class Visualization
. Besides the classical U-matrix, we can visualize the following:
An example of how to use the 2 classes can be found in the notebooks
folder.
Footnotes
-
The orginal formula is in reality much older, and can be referred to Kiviluoto, 1996 ↩