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

smile.classification.MLP 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.classification;

import java.io.Serial;
import java.io.Serializable;
import java.util.Arrays;
import java.util.Properties;

import smile.base.mlp.*;
import smile.math.MathEx;
import smile.util.IntSet;
import smile.util.Strings;

/**
 * Fully connected multilayer perceptron neural network for classification.
 * An MLP consists of at least three layers of nodes: an input layer,
 * a hidden layer and an output layer. The nodes are interconnected
 * through weighted acyclic arcs from each preceding layer to the
 * following, without lateral or feedback connections. Each node
 * calculates a transformed weighted linear combination of its inputs
 * (output activations from the preceding layer), with one of the weights
 * acting as a trainable bias connected to a constant input. The
 * transformation, called activation function, is a bounded non-decreasing
 * (non-linear) function.
 * 

* The representational capabilities of an MLP are determined by the range of * mappings it may implement through weight variation. Single layer perceptrons * are capable of solving only linearly separable problems. With the sigmoid * function as activation function, the single-layer network is identical * to the logistic regression model. *

* The universal approximation theorem for neural networks states that every * continuous function that maps intervals of real numbers to some output * interval of real numbers can be approximated arbitrarily closely by a * multi-layer perceptron with just one hidden layer. This result holds only * for restricted classes of activation functions, which are extremely complex * and NOT smooth for subtle mathematical reasons. On the other hand, smoothness * is important for gradient descent learning. Besides, the proof is not * constructive regarding the number of neurons required or the settings of * the weights. Therefore, complex systems will have more layers of neurons * with some having increased layers of input neurons and output neurons * in practice. *

* The most popular algorithm to train MLPs is back-propagation, which is a * gradient descent method. Based on chain rule, the algorithm propagates the * error back through the network and adjusts the weights of each connection in * order to reduce the value of the error function by some small amount. * For this reason, back-propagation can only be applied on networks with * differentiable activation functions. *

* During error back propagation, we usually times the gradient with a small * number η, called learning rate, which is carefully selected to ensure * that the network converges to a local minimum of the error function * fast enough, without producing oscillations. One way to avoid oscillation * at large η, is to make the change in weight dependent on the past weight * change by adding a momentum term. *

* Although the back-propagation algorithm may perform gradient * descent on the total error of all instances in a batch way, * the learning rule is often applied to each instance separately in an online * way or stochastic way. There exists empirical indication that the stochastic * way results in faster convergence. *

* In practice, the problem of over-fitting has emerged. This arises in * convoluted or over-specified systems when the capacity of the network * significantly exceeds the needed free parameters. There are two general * approaches for avoiding this problem: The first is to use cross-validation * and similar techniques to check for the presence of over-fitting and * optimally select hyperparameters such as to minimize the generalization * error. The second is to use some form of regularization, which emerges * naturally in a Bayesian framework, where the regularization can be * performed by selecting a larger prior probability over simpler models; * but also in statistical learning theory, where the goal is to minimize over * the "empirical risk" and the "structural risk". *

* For neural networks, the input patterns usually should be scaled/standardized. * Commonly, each input variable is scaled into interval [0, 1] or to have * mean 0 and standard deviation 1. *

* For penalty functions and output units, the following natural pairings are * recommended: *

    *
  • linear output units and a least squares penalty function. *
  • a two-class cross-entropy penalty function and a logistic * activation function. *
  • a multi-class cross-entropy penalty function and a softmax * activation function. *
* By assigning a softmax activation function on the output layer of * the neural network for categorical target variables, the outputs * can be interpreted as posterior probabilities, which are very useful. * * @author Haifeng Li */ public class MLP extends MultilayerPerceptron implements Classifier, Serializable { @Serial private static final long serialVersionUID = 2L; private static final org.slf4j.Logger logger = org.slf4j.LoggerFactory.getLogger(MLP.class); /** * The number of classes. */ private final int k; /** * The class label encoder. */ private final IntSet classes; /** * Constructor. * * @param builders the builders of layers from bottom to top. */ public MLP(LayerBuilder... builders) { super(net(builders)); int outSize = output.getOutputSize(); this.k = outSize == 1 ? 2 : outSize; this.classes = IntSet.of(k); } /** * Constructor. * * @param classes the class labels. * @param builders the builders of layers from bottom to top. */ public MLP(IntSet classes, LayerBuilder... builders) { super(net(builders)); int outSize = output.getOutputSize(); this.k = outSize == 1 ? 2 : outSize; this.classes = classes; } /** Builds the layers. */ private static Layer[] net(LayerBuilder... builders) { int p = 0; int l = builders.length; Layer[] net = new Layer[l]; for (int i = 0; i < l; i++) { net[i] = builders[i].build(p); p = builders[i].neurons(); } return net; } @Override public int numClasses() { return classes.size(); } @Override public int[] classes() { return classes.values; } @Override public int predict(double[] x, double[] posteriori) { propagate(x, false); int n = output.getOutputSize(); if (n == 1 && k == 2) { posteriori[1] = output.output()[0]; posteriori[0] = 1.0 - posteriori[1]; } else { System.arraycopy(output.output(), 0, posteriori, 0, n); } return classes.valueOf(MathEx.whichMax(posteriori)); } @Override public int predict(double[] x) { propagate(x, false); int n = output.getOutputSize(); if (n == 1 && k == 2) { return classes.valueOf(output.output()[0] > 0.5 ? 1 : 0); } else { return classes.valueOf(MathEx.whichMax(output.output())); } } @Override public boolean soft() { return true; } @Override public boolean online() { return true; } /** Updates the model with a single sample. RMSProp is not applied. */ @Override public void update(double[] x, int y) { propagate(x, true); setTarget(classes.indexOf(y)); backpropagate(true); t++; } /** Updates the model with a mini-batch. RMSProp is applied if {@code rho > 0}. */ @Override public void update(double[][] x, int[] y) { for (int i = 0; i < x.length; i++) { propagate(x[i], true); setTarget(classes.indexOf(y[i])); backpropagate(false); } update(x.length); t++; } /** Sets the network target vector. */ private void setTarget(int y) { int n = output.getOutputSize(); double t = output.cost() == Cost.LIKELIHOOD ? 1.0 : 0.9; double f = 1.0 - t; double[] target = this.target.get(); if (n == 1) { target[0] = y == 1 ? t : f; } else { Arrays.fill(target, f); target[y] = t; } } /** * Fits a MLP model. * @param x the training dataset. * @param y the training labels. * @param params the hyperparameters. * @return the model. */ public static MLP fit(double[][] x, int[] y, Properties params) { int p = x[0].length; int k = MathEx.max(y) + 1; LayerBuilder[] layers = Layer.of(k, p, params.getProperty("smile.mlp.layers", "ReLU(100)")); MLP model = new MLP(layers); model.setParameters(params); int epochs = Integer.parseInt(params.getProperty("smile.mlp.epochs", "100")); int batch = Integer.parseInt(params.getProperty("smile.mlp.mini_batch", "32")); double[][] batchx = new double[batch][]; int[] batchy = new int[batch]; for (int epoch = 1; epoch <= epochs; epoch++) { logger.info("{} epoch", Strings.ordinal(epoch)); int[] permutation = MathEx.permutate(x.length); for (int i = 0; i < x.length; i += batch) { int size = Math.min(batch, x.length - i); for (int j = 0; j < size; j++) { int index = permutation[i + j]; batchx[j] = x[index]; batchy[j] = y[index]; } if (size < batch) { model.update(Arrays.copyOf(batchx, size), Arrays.copyOf(batchy, size)); } else { model.update(batchx, batchy); } } } return model; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy