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

smile.base.rbf.RBF Maven / Gradle / Ivy

The newest version!
/*
 * 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.base.rbf;

import java.io.Serial;
import java.io.Serializable;
import java.util.Arrays;
import smile.clustering.CLARANS;
import smile.clustering.KMeans;
import smile.math.MathEx;
import smile.math.distance.Metric;
import smile.math.distance.EuclideanDistance;
import smile.math.rbf.RadialBasisFunction;
import smile.math.rbf.GaussianRadialBasis;

/**
 * A neuron in radial basis function network. A radial basis function network
 * is an artificial neural network that uses radial basis functions as
 * activation functions. It is a linear combination of radial basis functions.
 * 

* In its basic form, radial basis function network is in the form * * y(x) = Σ wi φ(||x-ci||) * * where the approximating function y(x) is represented as * a sum of N radial basis functions φ, each associated * with a different center ci, and weighted by an * appropriate coefficient wi. For distance, * one usually chooses Euclidean distance. The weights * wi can be estimated using the matrix methods of * linear least squares, because the approximating function is linear * in the weights. *

* The centers ci can be randomly selected * from training data, or learned by some clustering method (e.g. k-means), * or learned together with weight parameters undergo a supervised * learning processing (e.g. error-correction learning). * * @param the data type of samples. * * @author Haifeng Li */ public class RBF implements Serializable { @Serial private static final long serialVersionUID = 2L; /** The center of neuron. */ private final T center; /** Radial basis function. */ private final RadialBasisFunction rbf; /** Metric distance. */ private final Metric distance; /** * Constructor. * * @param center the center of neuron. * @param rbf the radial basis functions. * @param distance the distance metric functor. */ public RBF(T center, RadialBasisFunction rbf, Metric distance) { this.center = center; this.rbf = rbf; this.distance = distance; } /** * The activation function. * @param x the sample. * @return the activation function value. */ public double f(T x) { return rbf.f(distance.d(x, center)); } /** * Makes a set of RBF neurons. * * @param centers the neuron centers. * @param basis the radial basis functions. * @param distance the distance metric functor. * @param the data type of samples. * @return the RBF neurons. */ public static RBF[] of(T[] centers, RadialBasisFunction basis, Metric distance) { int k = centers.length; @SuppressWarnings("unchecked") RBF[] rbf = new RBF[k]; for (int i = 0; i < k; i++) { rbf[i] = new RBF<>(centers[i], basis, distance); } return rbf; } /** * Makes a set of RBF neurons. * * @param centers the neuron centers. * @param basis the radial basis functions. * @param distance the distance metric functor. * @param the data type of samples. * @return the RBF neurons. */ public static RBF[] of(T[] centers, RadialBasisFunction[] basis, Metric distance) { int k = centers.length; @SuppressWarnings("unchecked") RBF[] rbf = new RBF[k]; for (int i = 0; i < k; i++) { rbf[i] = new RBF<>(centers[i], basis[i], distance); } return rbf; } /** * Estimates the width of Gaussian RBF based on the maximum * distance between the chosen centers. The width is estimated * as dmax / sqrt(2*k), where k is number of centers. * This choice would be close to the optimal solution if the data * were uniformly distributed in the input space, leading to a * uniform distribution of centroids. */ private static double estimateWidth(T[] centers, Metric distance) { int k = centers.length; double r0 = 0.0; for (int i = 0; i < k; i++) { for (int j = 0; j < i; j++) { double d = distance.d(centers[i], centers[j]); if (r0 < d) { r0 = d; } } } r0 /= Math.sqrt(2*k); return r0; } /** * Estimates the width of Gaussian RBF based on the average distance to * nearest neighbors. * @param p the number of nearest neighbors of centers to estimate * the width of Gaussian RBF functions. */ private static double[] estimateWidth(T[] centers, Metric distance, int p) { int k = centers.length; double[] d = new double[k]; double[] r = new double[k]; for (int i = 0; i < k; i++) { for (int j = 0; j < k; j++) { d[j] = distance.d(centers[i], centers[j]); } Arrays.sort(d); double r0 = 0.0; for (int j = 1; j <= p; j++) { r0 += d[j]; } r[i] = r0 / p; } return r; } /** * Estimates the width of Gaussian RBF as the width of each * cluster multiplied with a given scaling parameter r. */ private static double[] estimateWidth(T[] x, int[] y, T[] centers, int[] clusterSize, Metric distance, double r) { int k = centers.length; double[] sigma = new double[k]; for (int i = 0; i < x.length; i++) { sigma[y[i]] += MathEx.pow2(distance.d(x[i], centers[y[i]])); } for (int i = 0; i < k; i++) { if (clusterSize[i] >= 5 || sigma[i] != 0.0) { sigma[i] = Math.sqrt(sigma[i] / clusterSize[i]); } else { sigma[i] = Double.POSITIVE_INFINITY; for (int j = 0; j < k; j++) { if (i != j) { double d = distance.d(centers[i], centers[j]); if (d < sigma[i]) { sigma[i] = d; } } } sigma[i] /= 2.0; } sigma[i] *= r; } return sigma; } /** Returns a set of Gaussian radial basis with given widths. */ private static GaussianRadialBasis[] gaussian(double[] width) { int k = width.length; GaussianRadialBasis[] basis = new GaussianRadialBasis[k]; for (int i = 0; i < k; i++) { basis[i] = new GaussianRadialBasis(width[i]); } return basis; } /** * Fits Gaussian RBF function and centers on data. The centers are * chosen as the centroids of K-Means. Let dmax be the maximum * distance between the chosen centers, the standard deviation (i.e. width) * of Gaussian radial basis function is dmax / sqrt(2*k), where * k is number of centers. This choice would be close to the optimal * solution if the data were uniformly distributed in the input space, * leading to a uniform distribution of centroids. * @param x the training dataset. * @param k the number of RBF neurons to learn. * @return a Gaussian RBF function with parameter learned from data. */ public static RBF[] fit(double[][] x, int k) { KMeans kmeans = KMeans.fit(x, k, 10, 1E-4); double[][] centers = kmeans.centroids; EuclideanDistance distance = new EuclideanDistance(); GaussianRadialBasis basis = new GaussianRadialBasis(estimateWidth(centers, distance)); return of(centers, basis, distance); } /** * Fits Gaussian RBF function and centers on data. The centers are * chosen as the centroids of K-Means. The standard deviation (i.e. width) * of Gaussian radial basis function is estimated by the p-nearest neighbors * (among centers, not all samples) heuristic. A suggested value for * p is 2. * @param x the training dataset. * @param k the number of RBF neurons to learn. * @param p the number of nearest neighbors of centers to estimate * the width of Gaussian RBF functions. * @return Gaussian RBF functions with parameter learned from data. */ public static RBF[] fit(double[][] x, int k, int p) { if (p < 1 || p >= k) { throw new IllegalArgumentException("Invalid number of nearest neighbors: " + p); } KMeans kmeans = KMeans.fit(x, k, 10, 1E-4); double[][] centers = kmeans.centroids; EuclideanDistance distance = new EuclideanDistance(); double[] width = estimateWidth(centers, distance, p); GaussianRadialBasis[] basis = gaussian(width); return of(centers, basis, distance); } /** * Fits Gaussian RBF function and centers on data. The centers are * chosen as the centroids of K-Means. The width * of Gaussian radial basis function is estimated as the width of each * cluster multiplied with a given scaling parameter r. * @param x the training dataset. * @param k the number of RBF neurons to learn. * @param r the scaling parameter. * @return Gaussian RBF functions with parameter learned from data. */ public static RBF[] fit(double[][] x, int k, double r) { if (r <= 0.0) { throw new IllegalArgumentException("Invalid scaling parameter: " + r); } KMeans kmeans = KMeans.fit(x, k, 10, 1E-4); double[][] centers = kmeans.centroids; EuclideanDistance distance = new EuclideanDistance(); double[] width = estimateWidth(x, kmeans.y, centers, kmeans.size, distance, r); GaussianRadialBasis[] basis = gaussian(width); return of(centers, basis, distance); } /** * Fits Gaussian RBF function and centers on data. The centers are * chosen as the medoids of CLARANS. Let dmax be the maximum * distance between the chosen centers, the standard deviation (i.e. width) * of Gaussian radial basis function is dmax / sqrt(2*k), where * k is number of centers. In this way, the radial basis functions are not * too peaked or too flat. This choice would be close to the optimal * solution if the data were uniformly distributed in the input space, * leading to a uniform distribution of medoids. * @param x the training dataset. * @param k the number of RBF neurons to learn. * @param distance the distance functor. * @param the data type of samples. * @return a Gaussian RBF function with parameter learned from data. */ public static RBF[] fit(T[] x, Metric distance, int k) { CLARANS clarans = CLARANS.fit(x, distance, k); T[] centers = clarans.centroids; GaussianRadialBasis basis = new GaussianRadialBasis(estimateWidth(centers, distance)); return of(centers, basis, distance); } /** * Fits Gaussian RBF function and centers on data. The centers are * chosen as the medoids of CLARANS. The standard deviation (i.e. width) * of Gaussian radial basis function is estimated by the p-nearest neighbors * (among centers, not all samples) heuristic. A suggested value for * p is 2. * @param x the training dataset. * @param distance the distance functor. * @param k the number of RBF neurons to learn. * @param p the number of nearest neighbors of centers to estimate the width * of Gaussian RBF functions. * @param the data type of samples. * @return Gaussian RBF functions with parameter learned from data. */ public static RBF[] fit(T[] x, Metric distance, int k, int p) { if (p < 1 || p >= k) { throw new IllegalArgumentException("Invalid number of nearest neighbors: " + p); } CLARANS clarans = CLARANS.fit(x, distance, k); T[] centers = clarans.centroids; double[] width = estimateWidth(centers, distance, p); GaussianRadialBasis[] basis = gaussian(width); return of(centers, basis, distance); } /** * Fits Gaussian RBF function and centers on data. The centers are * chosen as the medoids of CLARANS. The standard deviation (i.e. width) * of Gaussian radial basis function is estimated as the width of each * cluster multiplied with a given scaling parameter r. * @param x the training dataset. * @param k the number of RBF neurons to learn. * @param distance the distance functor. * @param r the scaling parameter. * @param the data type of samples. * @return Gaussian RBF functions with parameter learned from data. */ public static RBF[] fit(T[] x, Metric distance, int k, double r) { if (r <= 0.0) { throw new IllegalArgumentException("Invalid scaling parameter: " + r); } CLARANS clarans = CLARANS.fit(x, distance, k); T[] centers = clarans.centroids; double[] width = estimateWidth(x, clarans.y, centers, clarans.size, distance, r); GaussianRadialBasis[] basis = gaussian(width); return of(centers, basis, distance); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy