Implementing a self-organizing map, part 2: Nodes and the map
2025-03-11
With Vec
and Hex
in place, we're now ready to implement the SOM node.
Before we can construct a SOM we'll also need something to generate the map grid. I have a function here that generates a rectangular grid of Hex
s, but you could replace this with something that generates a funkier shape.
The SOM
Now we can get started on the meat of this exercise: implementing the SOM itself. First, let's create the Som
class with a constructor, a destructor and a getter for the vector that holds the nodes.
The SOM constructor accepts the numbers of rows and columns that the grid should contain. Since we will be using the weights of the nodes to define their color when they're drawn and we want the initial map to be made up of random colors, the components of the weights are drawn randomly from the range [0.0, 255.0). We could just as easily use any other range, such as [0.0, 1.0) for instance, which would make the initial map black.
We also set a training epoch limit, which limits the number of epochs that the map can be trained to 10. This is very low, but for learning colors it should be enough. The training epoch limit is used in calculating the learning rate.
Now we get to where the magic happens, the training method.
TrainOneEpoch()
takes a batch of training data vectors as input and uses them to train the map. The learning rate is calculated based on the current training epoch and the training epoch limit such that it decreases as the training progresses, which means that node weights are adjusted more drastically in the beginning. The BMU is sought for each training data sample, after which both are used to update the weights of all the nodes in the map. The neighborhood function used here is one that applies the update to the BMU in full and to other nodes based on their distance to the BMU such that each unit increase in the distance halves the update amount from the previous. A good alternative to this would be the Gaussian.
With all of this in place we now have a functioning SOM. Now all that's left is to build something around it that will allow us to observe how it works, which we'll do in part 3.