smile.vq.SOM Maven / Gradle / Ivy
/*
* Copyright (c) 2010-2021 Haifeng Li. All rights reserved.
*
* Smile is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Smile is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Smile. If not, see .
*/
package smile.vq;
import java.io.Serial;
import java.io.Serializable;
import java.util.Arrays;
import java.util.stream.IntStream;
import smile.clustering.CentroidClustering;
import smile.math.MathEx;
import smile.math.TimeFunction;
import smile.manifold.MDS;
import smile.sort.QuickSort;
/**
* Self-Organizing Map. An SOM is an unsupervised learning method to produce
* a low-dimensional (typically two-dimensional) discretized representation
* (called a map) of the input space of the training samples. The model was
* first described as an artificial neural network by Teuvo Kohonen, and is
* sometimes called a Kohonen map.
*
* While it is typical to consider SOMs as related to feed-forward networks where
* the nodes are visualized as being attached, this type of architecture is
* fundamentally different in arrangement and motivation because SOMs use a
* neighborhood function to preserve the topological properties of the input
* space. This makes SOMs useful for visualizing low-dimensional views of
* high-dimensional data, akin to multidimensional scaling.
*
* SOMs belong to a large family of competitive learning process and vector
* quantization. An SOM consists of components called nodes or neurons.
* Associated with each node is a weight vector of the same dimension as
* the input data vectors and a position in the map space. The usual arrangement
* of nodes is a regular spacing in a hexagonal or rectangular grid. The
* self-organizing map describes a mapping from a higher dimensional input
* space to a lower dimensional map space. During the (iterative) learning,
* the input vectors are compared to the weight vector of each neuron. Neurons
* who most closely match the input are known as the best match unit (BMU) of
* the system. The weight vector of the BMU and those of nearby neurons are
* adjusted to be closer to the input vector by a certain step size.
*
* There are two ways to interpret a SOM. Because in the training phase weights
* of the whole neighborhood are moved in the same direction, similar items
* tend to excite adjacent neurons. Therefore, SOM forms a semantic map where
* similar samples are mapped close together and dissimilar apart.
* The other way is to think of neuronal weights as pointers to the input space.
* They form a discrete approximation of the distribution of training samples.
* More neurons point to regions with high training sample concentration and
* fewer where the samples are scarce.
*
* SOM may be considered a nonlinear generalization of Principal components
* analysis (PCA). It has been shown, using both artificial and real
* geophysical data, that SOM has many advantages over the conventional feature
* extraction methods such as Empirical Orthogonal Functions (EOF) or PCA.
*
* It has been shown that while SOMs with a small number of nodes behave in a
* way that is similar to K-means. However, larger SOMs rearrange data
* in a way that is fundamentally topological in character and display properties
* which are emergent. Therefore, large maps are preferable to smaller ones.
* In maps consisting of thousands of nodes, it is possible to perform cluster
* operations on the map itself.
*
* A common way to display SOMs is the heat map of U-matrix. The U-matrix value
* of a particular node is the minimum/maximum/average distance between the node
* and its closest neighbors. In a rectangular grid for instance, we might
* consider the closest 4 or 8 nodes.
*
*
References
*
* - Teuvo KohonenDan. Self-organizing maps. Springer, 3rd edition, 2000.
*
*
* @author Haifeng Li
*/
public class SOM implements VectorQuantizer {
@Serial
private static final long serialVersionUID = 2L;
/**
* Self-Organizing Map Neuron.
*/
private static class Neuron implements Serializable {
/** The weight vector. */
public final double[] w;
/** The row index of neuron in the lattice. */
public final int i;
/** The column index of neuron in the lattice. */
public final int j;
/**
* Constructor.
* @param i the row index of neuron in the lattice.
* @param j the column index of neuron in the lattice.
* @param w the weight vector.
*/
public Neuron(int i, int j, double[] w) {
this.i = i;
this.j = j;
this.w = w;
}
}
/**
* The number of rows in the lattice.
*/
private final int nrow;
/**
* The number of columns in the lattice.
*/
private final int ncol;
/**
* The lattice of neurons.
*/
private final Neuron[][] map;
/**
* The neurons in linear array.
*/
private final Neuron[] neurons;
/**
* The distance between a new observation to neurons.
*/
private final double[] dist;
/**
* The learning rate function.
*/
private final TimeFunction alpha;
/**
* The neighborhood function.
*/
private final Neighborhood theta;
/**
* The threshold to update neuron if {@code alpha * theta > eps}.
*/
private final double tol = 1E-5;
/**
* The current iteration.
*/
private int t = 0;
/**
* Constructor.
* @param neurons the initial lattice of neurons.
* @param alpha the learning rate function.
* @param theta the neighborhood function.
*/
public SOM(double[][][] neurons, TimeFunction alpha, Neighborhood theta) {
this.alpha = alpha;
this.theta = theta;
this.nrow = neurons.length;
this.ncol = neurons[0].length;
this.map = new Neuron[nrow][ncol];
this.neurons = new Neuron[nrow * ncol];
this.dist = new double[this.neurons.length];
for (int i = 0, k = 0; i < nrow; i++) {
for (int j = 0; j < ncol; j++, k++) {
Neuron neuron = new Neuron(i, j, neurons[i][j].clone());
map[i][j] = neuron;
this.neurons[k] = neuron;
}
}
}
/**
* Creates a lattice of which the weight vectors are randomly selected from samples.
* @param nrow the number of rows in the lattice.
* @param ncol the number of columns in the lattice.
* @param samples the samples to draw initial weight vectors.
* @return the lattice.
*/
public static double[][][] lattice(int nrow, int ncol, double[][] samples) {
int k = nrow * ncol;
int n = samples.length;
int[] clusters = new int[n];
double[][] medoids = new double[k][];
CentroidClustering.seed(samples, medoids, clusters, MathEx::squaredDistance);
// Pair-wise distance matrix.
double[][] pdist = new double[k][k];
MathEx.pdist(medoids, pdist, MathEx::distance);
MDS mds = MDS.of(pdist);
double[][] coordinates = mds.coordinates;
double[] x = Arrays.stream(coordinates).mapToDouble(point -> point[0]).toArray();
double[] y = new double[ncol];
int[] row = new int[ncol];
int[] index = QuickSort.sort(x);
double[][][] neurons = new double[nrow][ncol][];
for (int i = 0; i < nrow; i++) {
for (int j = 0; j < ncol; j++) {
int point = index[i * ncol + j];
y[j] = coordinates[point][1];
row[j] = point;
}
QuickSort.sort(y, row);
for (int j = 0; j < ncol; j++) {
neurons[i][j] = medoids[row[j]];
}
}
return neurons;
}
@Override
public void update(double[] x) {
Neuron bmu = bmu(x);
int i = bmu.i;
int j = bmu.j;
int d = bmu.w.length;
double alpha = this.alpha.apply(t);
Arrays.stream(neurons).parallel().forEach(neuron -> {
double delta = alpha * theta.of(neuron.i - i, neuron.j - j, t);
if (delta > tol) {
double[] w = neuron.w;
for (int k = 0; k < d; k++) {
w[k] += delta * (x[k] - w[k]);
}
}
});
t = t + 1;
}
/**
* Returns the lattice of neurons.
* @return the lattice of neurons.
*/
public double[][][] neurons() {
double[][][] lattice = new double[nrow][ncol][];
for (int i = 0; i < nrow; i++) {
for (int j = 0; j < ncol; j++) {
lattice[i][j] = map[i][j].w;
}
}
return lattice;
}
/**
* Calculates the unified distance matrix (u-matrix) for visualization.
* U-matrix is a popular method of displaying SOMs. The value of umatrix
* is the maximum of distances between a map unit to its neighbors.
*
* @return the unified distance matrix.
*/
public double[][] umatrix() {
double[][] umatrix = new double[nrow][ncol];
for (int i = 0; i < nrow - 1; i++) {
for (int j = 0; j < ncol - 1; j++) {
double dist = Math.sqrt(MathEx.distance(map[i][j].w, map[i][j + 1].w));
umatrix[i][j] = Math.max(umatrix[i][j], dist);
umatrix[i][j + 1] = Math.max(umatrix[i][j + 1], dist);
dist = Math.sqrt(MathEx.distance(map[i][j].w, map[i + 1][j].w));
umatrix[i][j] = Math.max(umatrix[i][j], dist);
umatrix[i + 1][j] = Math.max(umatrix[i + 1][j], dist);
}
}
for (int i = 0; i < nrow - 1; i++) {
double dist = Math.sqrt(MathEx.distance(map[i][ncol - 1].w, map[i + 1][ncol - 1].w));
umatrix[i][ncol - 1] = Math.max(umatrix[i][ncol - 1], dist);
umatrix[i + 1][ncol - 1] = Math.max(umatrix[i + 1][ncol - 1], dist);
}
for (int j = 0; j < ncol - 1; j++) {
double dist = Math.sqrt(MathEx.distance(map[nrow - 1][j].w, map[nrow - 1][j + 1].w));
umatrix[nrow - 1][j] = Math.max(umatrix[nrow - 1][j], dist);
umatrix[nrow - 1][j + 1] = Math.max(umatrix[nrow - 1][j + 1], dist);
}
umatrix[nrow - 1][ncol - 1] = Math.max(umatrix[nrow - 1][ncol - 2], umatrix[nrow - 2][ncol - 1]);
return umatrix;
}
@Override
public double[] quantize(double[] x) {
return bmu(x).w;
}
/** Returns the best matching unit. */
private Neuron bmu(double[] x) {
IntStream.range(0, neurons.length).parallel().forEach(i -> dist[i] = MathEx.distance(neurons[i].w, x));
QuickSort.sort(dist, neurons);
return neurons[0];
}
}