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

smile.feature.extraction.GHA 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.feature.extraction;

import smile.data.Tuple;
import smile.data.DataFrame;
import smile.math.MathEx;
import smile.math.TimeFunction;
import smile.math.matrix.Matrix;

import java.io.Serial;

/**
 * Generalized Hebbian Algorithm. GHA is a linear feed-forward neural
 * network model for unsupervised learning with applications primarily in
 * principal components analysis. It is single-layer process -- that is, a
 * synaptic weight changes only depending on the response of the inputs and
 * outputs of that layer.
 * 

* It guarantees that GHA finds the first k eigenvectors of the covariance matrix, * assuming that the associated eigenvalues are distinct. The convergence theorem * is formulated in terms of a time-varying learning rate η. In practice, the * learning rate η is chosen to be a small constant, in which case convergence is * guaranteed with mean-squared error in synaptic weights of order η. *

* It also has a simple and predictable trade-off between learning speed and * accuracy of convergence as set by the learning rate parameter η. It was * shown that a larger learning rate η leads to faster convergence * and larger asymptotic mean-square error, which is intuitively satisfying. *

* Compared to regular batch PCA algorithm based on eigen decomposition, GHA is * an adaptive method and works with an arbitrarily large sample size. The storage * requirement is modest. Another attractive feature is that, in a non-stationary * environment, it has an inherent ability to track gradual changes in the * optimal solution in an inexpensive way. * *

References

*
    *
  1. Terence D. Sanger. Optimal unsupervised learning in a single-layer linear feedforward neural network. Neural Networks 2(6):459-473, 1989.
  2. *
  3. Simon Haykin. Neural Networks: A Comprehensive Foundation (2 ed.). 1998.
  4. *
* * @see PCA * * @author Haifeng Li */ public class GHA extends Projection { @Serial private static final long serialVersionUID = 2L; /** * The dimension of feature space. */ private final int p; /** * The dimension of input space. */ private final int n; /** * The learning rate; */ private final TimeFunction r; /** * Workspace for W * x. */ private final double[] y; /** * Workspace for W' * y. */ private final double[] wy; /** * The training iterations. */ protected int t = 0; /** * Constructor. * @param n the dimension of input space. * @param p the dimension of feature space. * @param r the learning rate. * @param columns the columns to transform when applied on Tuple/DataFrame. */ public GHA(int n, int p, TimeFunction r, String... columns) { super(new Matrix(p, n), "GHA", columns); if (n < 2) { throw new IllegalArgumentException("Invalid dimension of input space: " + n); } if (p < 1 || p > n) { throw new IllegalArgumentException("Invalid dimension of feature space: " + p); } this.n = n; this.p = p; this.r = r; y = new double[p]; wy = new double[n]; for (int i = 0; i < p; i++) { for (int j = 0; j < n; j++) { projection.set(i, j, 0.1 * MathEx.random()); } } } /** * Constructor. * @param w the initial projection matrix. When GHA converges, * the column of projection matrix are the first p * eigenvectors of covariance matrix, ordered by * decreasing eigenvalues. * @param r the learning rate. * @param columns the columns to transform when applied on Tuple/DataFrame. */ public GHA(double[][] w, TimeFunction r, String... columns) { super(Matrix.of(w), "GHA", columns); this.p = w.length; this.n = w[0].length; this.r = r; y = new double[p]; wy = new double[n]; } /** * Update the model with a new sample. * @param x the centered learning sample whose E(x) = 0. * @return the approximation error for input sample. */ public double update(double[] x) { if (x.length != n) { throw new IllegalArgumentException(String.format("Invalid input vector size: %d, expected: %d", x.length, n)); } projection.mv(x, y); for (int j = 0; j < p; j++) { for (int i = 0; i < n; i++) { double delta = x[i]; for (int l = 0; l <= j; l++) { delta -= projection.get(l, i) * y[l]; } projection.add(j, i, r.apply(t) * y[j] * delta); if (Double.isInfinite(projection.get(j, i))) { throw new IllegalStateException("GHA lost convergence. Lower learning rate?"); } } } t++; projection.mv(x, y); projection.tv(y, wy); return MathEx.squaredDistance(x, wy); } /** * Update the model with a new sample. * @param x the centered learning sample whose E(x) = 0. * @return the approximation error for input sample. */ public double update(Tuple x) { return update(x.toArray(columns)); } /** * Update the model with a set of samples. * @param data the centered learning samples whose E(x) = 0. */ public void update(double[][] data) { for (double[] x : data) { update(x); } } /** * Update the model with a new data frame. * @param data the centered learning samples whose E(x) = 0. */ public void update(DataFrame data) { update(data.toArray(columns)); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy