All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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

*
    *
  1. Teuvo KohonenDan. Self-organizing maps. Springer, 3rd edition, 2000.
  2. *
* * @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]; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy