smile.classification.AdaBoost 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.classification;
import java.io.Serial;
import java.util.Arrays;
import java.util.Properties;
import smile.base.cart.CART;
import smile.base.cart.SplitRule;
import smile.data.DataFrame;
import smile.data.Tuple;
import smile.data.formula.Formula;
import smile.data.type.StructType;
import smile.data.vector.BaseVector;
import smile.feature.importance.TreeSHAP;
import smile.math.MathEx;
import smile.util.IntSet;
import smile.util.Strings;
/**
* AdaBoost (Adaptive Boosting) classifier with decision trees. In principle,
* AdaBoost is a meta-algorithm, and can be used in conjunction with many other
* learning algorithms to improve their performance. In practice, AdaBoost with
* decision trees is probably the most popular combination. AdaBoost is adaptive
* in the sense that subsequent classifiers built are tweaked in favor of those
* instances misclassified by previous classifiers. AdaBoost is sensitive to
* noisy data and outliers. However, in some problems it can be less susceptible
* to the over-fitting problem than most learning algorithms.
*
* AdaBoost calls a weak classifier repeatedly in a series of rounds from
* total T classifiers. For each call a distribution of weights is updated
* that indicates the importance of examples in the data set for the
* classification. On each round, the weights of each incorrectly classified
* example are increased (or alternatively, the weights of each correctly
* classified example are decreased), so that the new classifier focuses more
* on those examples.
*
* The basic AdaBoost algorithm is only for binary classification problem.
* For multi-class classification, a common approach is reducing the
* multi-class classification problem to multiple two-class problems.
* This implementation is a multi-class AdaBoost without such reductions.
*
*
References
*
* - Yoav Freund, Robert E. Schapire. A Decision-Theoretic Generalization of on-Line Learning and an Application to Boosting, 1995.
* - Ji Zhu, Hui Zhou, Saharon Rosset and Trevor Hastie. Multi-class Adaboost, 2009.
*
*
* @author Haifeng Li
*/
public class AdaBoost extends AbstractClassifier implements DataFrameClassifier, TreeSHAP {
@Serial
private static final long serialVersionUID = 2L;
private static final org.slf4j.Logger logger = org.slf4j.LoggerFactory.getLogger(AdaBoost.class);
/**
* The model formula.
*/
private final Formula formula;
/**
* The number of classes.
*/
private final int k;
/**
* Forest of decision trees.
*/
private DecisionTree[] trees;
/**
* The weight of each decision tree.
*/
private double[] alpha;
/**
* The weighted error of each decision tree during training.
*/
private double[] error;
/**
* Variable importance. Every time a split of a node is made on variable
* the (GINI, information gain, etc.) impurity criterion for the two
* descendent nodes is less than the parent node. Adding up the decreases
* for each individual variable over all trees in the forest gives a fast
* variable importance that is often very consistent with the permutation
* importance measure.
*/
private final double[] importance;
/**
* Constructor.
*
* @param formula a symbolic description of the model to be fitted.
* @param k the number of classes.
* @param trees forest of decision trees.
* @param alpha the weight of each decision tree.
* @param error the weighted error of each decision tree during training.
* @param importance variable importance
*/
public AdaBoost(Formula formula, int k, DecisionTree[] trees, double[] alpha, double[] error, double[] importance) {
this(formula, k, trees, alpha, error, importance, IntSet.of(k));
}
/**
* Constructor.
*
* @param formula a symbolic description of the model to be fitted.
* @param k the number of classes.
* @param trees forest of decision trees.
* @param alpha the weight of each decision tree.
* @param error the weighted error of each decision tree during training.
* @param importance variable importance
* @param labels the class label encoder.
*/
public AdaBoost(Formula formula, int k, DecisionTree[] trees, double[] alpha, double[] error, double[] importance, IntSet labels) {
super(labels);
this.formula = formula;
this.k = k;
this.trees = trees;
this.alpha = alpha;
this.error = error;
this.importance = importance;
}
/**
* Fits a AdaBoost model.
*
* @param formula a symbolic description of the model to be fitted.
* @param data the data frame of the explanatory and response variables.
* @return the model.
*/
public static AdaBoost fit(Formula formula, DataFrame data) {
return fit(formula, data, new Properties());
}
/**
* Fits a AdaBoost model.
*
* @param formula a symbolic description of the model to be fitted.
* @param data the data frame of the explanatory and response variables.
* @param params the hyperparameters.
* @return the model.
*/
public static AdaBoost fit(Formula formula, DataFrame data, Properties params) {
int ntrees = Integer.parseInt(params.getProperty("smile.adaboost.trees", "500"));
int maxDepth = Integer.parseInt(params.getProperty("smile.adaboost.max_depth", "20"));
int maxNodes = Integer.parseInt(params.getProperty("smile.adaboost.max_nodes", "6"));
int nodeSize = Integer.parseInt(params.getProperty("smile.adaboost.node_size", "1"));
return fit(formula, data, ntrees, maxDepth, maxNodes, nodeSize);
}
/**
* Fits a AdaBoost model.
*
* @param formula a symbolic description of the model to be fitted.
* @param data the data frame of the explanatory and response variables.
* @param ntrees the number of trees.
* @param maxDepth the maximum depth of the tree.
* @param maxNodes the maximum number of leaf nodes in the tree.
* @param nodeSize the number of instances in a node below which the tree will
* not split, setting nodeSize = 5 generally gives good results.
* @return the model.
*/
public static AdaBoost fit(Formula formula, DataFrame data, int ntrees, int maxDepth, int maxNodes, int nodeSize) {
if (ntrees < 1) {
throw new IllegalArgumentException("Invalid number of trees: " + ntrees);
}
formula = formula.expand(data.schema());
DataFrame x = formula.x(data);
BaseVector, ?, ?> y = formula.y(data);
ClassLabels codec = ClassLabels.fit(y);
int[][] order = CART.order(x);
int k = codec.k;
int n = data.size();
int[] samples = new int[n];
double[] w = new double[n];
boolean[] wrong = new boolean[n];
Arrays.fill(w, 1.0);
double guess = 1.0 / k; // accuracy of random guess.
double b = Math.log(k - 1); // the bias to tree weight in case of multi-class.
int failures = 0; // the number of weak classifiers less accurate than guess.
DecisionTree[] trees = new DecisionTree[ntrees];
double[] alpha = new double[ntrees];
double[] error = new double[ntrees];
for (int t = 0; t < ntrees; t++) {
double W = MathEx.sum(w);
for (int i = 0; i < n; i++) {
w[i] /= W;
}
Arrays.fill(samples, 0);
int[] rand = MathEx.random(w, n);
for (int s : rand) {
samples[s]++;
}
trees[t] = new DecisionTree(x, codec.y, y.field(), k, SplitRule.GINI, maxDepth, maxNodes, nodeSize, -1, samples, order);
for (int i = 0; i < n; i++) {
wrong[i] = trees[t].predict(x.get(i)) != codec.y[i];
}
double e = 0.0; // weighted error
for (int i = 0; i < n; i++) {
if (wrong[i]) {
e += w[i];
}
}
logger.info("Training {} tree, weighted error = {}%", Strings.ordinal(t+1), 100*e);
if (1 - e > guess) {
failures = 0;
} else {
logger.error("Skip the weak classifier");
if (++failures > 3) {
trees = Arrays.copyOf(trees, t);
alpha = Arrays.copyOf(alpha, t);
error = Arrays.copyOf(error, t);
break;
} else {
t--;
continue;
}
}
error[t] = e;
alpha[t] = Math.log((1-e) / Math.max(1E-10, e)) + b;
double a = Math.exp(alpha[t]);
for (int i = 0; i < n; i++) {
if (wrong[i]) {
w[i] *= a;
}
}
}
double[] importance = new double[x.ncol()];
for (DecisionTree tree : trees) {
double[] imp = tree.importance();
for (int i = 0; i < imp.length; i++) {
importance[i] += imp[i];
}
}
return new AdaBoost(formula, k, trees, alpha, error, importance, codec.classes);
}
@Override
public Formula formula() {
return formula;
}
@Override
public StructType schema() {
return trees[0].schema();
}
/**
* Returns the variable importance. Every time a split of a node is made
* on variable the (GINI, information gain, etc.) impurity criterion for
* the two descendent nodes is less than the parent node. Adding up the
* decreases for each individual variable over all trees in the forest
* gives a simple measure of variable importance.
*
* @return the variable importance
*/
public double[] importance() {
return importance;
}
/**
* Returns the number of trees in the model.
*
* @return the number of trees in the model
*/
public int size() {
return trees.length;
}
/**
* Returns the decision trees.
* @return the decision trees.
*/
public DecisionTree[] trees() {
return trees;
}
/**
* Trims the tree model set to a smaller size in case of over-fitting.
* Or if extra decision trees in the model don't improve the performance,
* we may remove them to reduce the model size and also improve the speed of
* prediction.
*
* @param ntrees the new (smaller) size of tree model set.
*/
public void trim(int ntrees) {
if (ntrees > trees.length) {
throw new IllegalArgumentException("The new model size is larger than the current size.");
}
if (ntrees <= 0) {
throw new IllegalArgumentException("Invalid new model size: " + ntrees);
}
if (ntrees < trees.length) {
trees = Arrays.copyOf(trees, ntrees);
alpha = Arrays.copyOf(alpha, ntrees);
error = Arrays.copyOf(error, ntrees);
}
}
@Override
public boolean soft() {
return true;
}
@Override
public int predict(Tuple x) {
Tuple xt = formula.x(x);
double[] y = new double[k];
for (int i = 0; i < trees.length; i++) {
y[trees[i].predict(xt)] += alpha[i];
}
return classes.valueOf(MathEx.whichMax(y));
}
/**
* Predicts the class label of an instance and also calculate a posteriori
* probabilities. Not supported.
*/
@Override
public int predict(Tuple x, double[] posteriori) {
Tuple xt = formula.x(x);
Arrays.fill(posteriori, 0.0);
for (int i = 0; i < trees.length; i++) {
posteriori[trees[i].predict(xt)] += alpha[i];
}
double sum = MathEx.sum(posteriori);
for (int i = 0; i < k; i++) {
posteriori[i] /= sum;
}
return classes.valueOf(MathEx.whichMax(posteriori));
}
/**
* Test the model on a validation dataset.
* @param data the validation data.
* @return the predictions with first 1, 2, ..., decision trees.
*/
public int[][] test(DataFrame data) {
DataFrame x = formula.x(data);
int n = x.size();
int ntrees = trees.length;
int[][] prediction = new int[ntrees][n];
if (k == 2) {
for (int j = 0; j < n; j++) {
Tuple xj = x.get(j);
double base = 0;
for (int i = 0; i < ntrees; i++) {
base += alpha[i] * trees[i].predict(xj);
prediction[i][j] = base > 0 ? 1 : 0;
}
}
} else {
double[] p = new double[k];
for (int j = 0; j < n; j++) {
Tuple xj = x.get(j);
Arrays.fill(p, 0);
for (int i = 0; i < ntrees; i++) {
p[trees[i].predict(xj)] += alpha[i];
prediction[i][j] = MathEx.whichMax(p);
}
}
}
return prediction;
}
}