[关闭]
@Dounm 2018-08-25T12:06:29.000000Z 字数 11576 阅读 18500

GBDT算法详解

MachineLearning


GBDT(Gradient Boosting Descision Tree),是各类机器学习算法比赛的赢家们都非常青睐的算法,也是效果最好的统计学习方法之一。

本文将从决策树和Boosting算法的基础知识开始,详细的推导GBDT的原理。如果不足,还望互相讨论。

1. 机器学习基础

依照《统计学习方法》一书所说,统计学习的三要素是:模型+策略+算法

针对监督学习而言,

2. 决策树

决策树是一种基本的分类和回归方法,它的模型呈树形结构。
对于决策树,我们可以认为它是定义在特征空间和类空间上的条件概率分布

以分类决策树为例,决策树的每条路径对应划分中的一个单元(cell),在每个单元上定义一个类的概率分布就构成了一个条件概率分布。

表示特征的随机变量,表示类的随机变量,则这个条件概率分布就是取值自单元的集合,取值自类的集合。通常各叶节点(单元)上的条件概率分布往往偏向于某个类,决策树分类时会将该节点的实例强行分到条件概率大的那一类。

2.1 决策树学习

决策树学习的本质是从训练数据集中归纳出一组分类规则。能正确分类训练数据集的决策树可能有多个,也可能一个没有。我们需要一个分类错误少,且具有良好泛化能力的决策树。

由于从所有可能决策树中选择最优决策树是NP完全问题,所以现实生活中决策树学习算法通常采用启发式方法,近似求解最优化问题。这样得到的决策树是次最优sub-optimal

决策树的学习包括三个过程:

  1. 特征选择
  2. 决策树的生成(考虑局部最优)
  3. 决策树的剪枝(考虑全局最优)
    1. 预剪枝(Pre-Pruning):在决策树生成同时进行剪枝
    2. 后剪枝(Post-Pruning):在决策树构造完成后进行剪枝(通常后剪枝效果好于预剪枝,因此后面该节讨论的都是后剪枝)

决策树生成的具体步骤如下:

  1. 选择一个最优特征(使得各个子集有一个在当前条件下最好的分类),然后按照该特征划分数据集。
  2. 如果这些子集能够基本正确分类,则构建叶节点
  3. 否则,对这些子集继续选择新的最优特征,然后继续分割
  4. 循环往复,直到所有训练数据子集已经被基本正确分类,或没有合适的特征为止

可知,在生成决策树时,我们采取的是【贪心】的方法。

对于上述生成步骤生成的决策树,对于训练数据拟合会很好,但是泛化能力却很差,有过拟合的问题,因此需要剪枝步骤来防止过拟合。

而决策树的剪枝步骤,则是通过极小化决策树的整体损失函数(loss function)来实现的。常见损失函数的形式为

而剪枝,就是在确定时,选择令损失函数最小的模型。具体来说,就是通过递归的从树的叶子结点往上缩,判断剪枝后的树是否比未剪枝的树的值小,小则剪枝成功。

2.2 CART: Classification and Regression Tree

CART树假定决策树是二叉树,内部节点特征取值是“是”或“否”。这就等价于递归的二分每个特征,将输入空间即特征空间划分为有限个单元,在这些单元上预测概率分布。

CART的算法步骤也分为两步,分别是决策树生成和决策树剪枝。

在决策树生成时,CART算法要求生成尽可能大的决策树。对回归树来说,通过最小化某个函数来选择最优切分变量和最优切分值;分类树则以基尼指数来选择最有特征和最优二值切分点。

决策树剪枝时,其思想和普通决策树类似,区别选择额外的交叉验证集来选择最终的最优决策树[1]

2.2.1 最小二乘回归树的生成

以最小二乘回归树(least squares regression tree)为例,我们来了解下决策树的生成步骤。

设数据集为,设树已有个叶子结点,每个叶子结点有个固定的输出值,那么决策树模型就可表示为

最小二乘回归树使用平方误差来表示预测误差,我们需要求解出每个叶子节点上的最优输出值。对于平方误差求解最小值,易得

关键在于如何对输入空间进行划分?我们需要寻找最优切分变量和最优切分点。即求解

我们需要遍历所有变量,然后固定遍历切分点,使得上式达到最小。

不断地对划分后的子空间按上述方法进行递归划分,直到满足停止条件(例如每个叶子结点仅有一个样本)。

3. Boosting提升方法

提升方法是一种非常常见的统计学习方法,通常的思路是改变训练数据的概率分布(训练数据的权值分布),针对不同训练数据分布调用弱学习方法学习一系列的弱分类器

因此针对于Boosting方法,就有两个地方需要注意:

  1. 每一轮如何改变训练数据的权值和概率分布
  2. 如何将弱分类器组合成一个强分类器

以Boosting算法里常见的AdaBoost算法为例,对于第1个问题,AdaBoost的方法是提升被前一轮弱分类器错误分类的样本的权值,降低被正确分类样本的权值;对于第2个问题,AdaBoost采取加权多数表决的方法,加大分类误差率小的弱分类器的权值,减少分类误差率大的弱分类器的权值。

3.1 加法模型与前向分步算法

针对前面两个问题,大部分Boosting算法都是用加法模型(additive model)来组合弱分类器,然后用前向分步算法(forward stagewise algorithm)来训练改变训练数据的权值和概率分布(包括AdaBoost算法)。

考虑加法模型

在给定训练数据以及损失函数的条件下,学习加法模型就是一个经验风险极小化问题即损失函数极小化问题

直接求解这个的最小值是个很复杂的优化问题,因此我们采用前向分步算法来解决这一优化问题。

前向分步算法的思路是:因为学习的是加法模型,若能从前到后,每一步只学习一个基函数及其系数,逐步逼近优化目标函数,那么就可以简化复杂度。每步只需要优化如下损失函数

这样的话,原本需要同时求解从所有参数的优化问题就简化为逐次求解的优化问题。

前向分步算法的步骤为
1
(图片来自《统计学习方法》 8.3.1)

3.2 Boosting Tree提升树

Boosting方法可以以多种模型作为基本模型,其中以分类决策树或回归决策树为基模型的话被称为提升树(Boosting Tree),是统计学习中性能最好的方法之一[2]

按照第1节的说法,提升树实际上采用的是

以平方损失函数和回归决策树为例,推导下提升树的算法。

首先,提升树模型其实就是决策树的加法模型,表现为:

[3]

回归问题提升树的前向分步算法如下:

在前向分步算法的第步,给定当前模型,需要求解

从而得到,即第棵树的参数。

当采用平方误差损失函数时,其损失变为

所以,对于回归问题的Boosting Tree来说,每一步只需要拟合当前模型的残差即可。

3.3 Gradient Boosting梯度提升

提升树利用加法模型和前向分步算法实现优化过程时,当损失函数是平方损失函数时,每一步的优化很简单。但对于一般损失函数而言,往往每一步的优化没那么简单,所以引入了梯度提升(Gradient Boosting)算法。

梯度提升法利用损失函数的负梯度在当前模型的值作为之前的残差的近似值,拟合一个回归树。

对于任意可导函数来说,是必然情况。
针对于,它的自变量是,所以沿着负梯度方向更新是合理的。

算法如下:
2
3
(图片来自《统计学习方法》 8.4.3)

4. GBDT(Gradient Boosting Decision Tree)和 Xgboost

GBDT,Gradient Boosting Descision Tree,就是前面所提到的Gradient Boosting与Boosting Tree结合的结果。GBDT所采用的也是加法模型前向分步算法,树的类型则是CART树,loss函数不定。

4.1 GBDT的目标函数

对于普通的机器学习模型而言,其目标函数可以定义为如下:

其中,前面的是函数,后面的是正则化项。

结合前述的前向分步算法的原理,在第步时,目标函数就是

此时最优化该目标函数,就求得了

4.1.2 负梯度的理论支撑

前面第3.3节提到Gradient Boosting时,提及Gradient Boosting以负梯度代替残差来求解基函数,实际上,负梯度的理论支撑则是泰勒公式的一阶展开。即

作泰勒一阶展开,得到

此时公式目标函数(不考虑正则项)则变成

我们肯定希望函数每步都减小的,即,那么关键就在于这一项了。因为我们不知道到底是正还是负,那么只需让是我们任取的一个正系数)就能让一直恒为负了。

4.1.2 xgboost的目标函数

xgboost是GBDT的一个变种,也是最广泛的一种应用。它在对函数进行泰勒展开时,取的是二阶展开而非一阶展开,因此与实际函数的值更接近(前提条件要求函数二阶可导)。

泰勒公式的二阶展开为:

带入到公式中,则

因为在第步时,是已知值,所以是常数,不影响函数优化,可以省去,则

对这个Obj进行优化得到的就是第步的,然后最终将每一步的加在一起就得到了整体模型。

后续章节中,我们都使用xgboost的目标函数来进行推导。

4.2 用决策树来表示

对于决策树来说,设它有个叶子结点(即第二节提到的路径单元cell),那么每个叶子结点都有固定的值,且每个样本都必然存在且仅存在于一个叶子节点上,因此

另一方面,决策树的复杂度可以由来定义,即决策树叶子结点的数量和叶子结点取值的L2范数。

因此,假设为第个叶子结点的样本集合,那么公式就变为如下形式:

当位于第步时,在树的结构固定的情况下,我们已经知道每个叶子结点有哪些样本,则是确定的。又因为步的导数,所以也是固定的,因此都是固定的。

函数关于的一阶导为,即可求得最优的函数对于来说,是个凸函数),即

注意,二阶泰勒展开的函数针对于是凸函数,因此可以直接求出最优解析解。而一阶展开时的函数不是可导的,所以只能逐步降低函数。(估计这也是XGBoost选择二阶泰勒展开的原因之一)

则针对于结构固定的决策树,最优的函数即为:

4.3 求解最优结构的决策树

前面提到,对于固定结构的决策树,我们可以得知其最优的函数的值。那么,该如何求解最优结构的决策树呢?

节提到,决策树通常采用后剪枝的方案,因此其学习分为两个阶段:决策树的生成(特征选择在生成阶段处理)和决策树的剪枝。其中决策树的生成阶段仅考虑如何更好的对训练数据进行拟合,只有到了决策树的剪枝阶段才考虑到了减少模型复杂度的问题。

在普通的GBDT采用决策树是CART,因此也是用后剪枝来处理过拟合的问题。

但是在xgboost中用的却是预剪枝。在函数中,我们既考虑了用拟合训练数据,又考虑了用正则项来减少模型复杂度。因此xgboost的在决策树的生成阶段就处理了过拟合的问题,无需独立的剪枝阶段。

具体步骤如下:

  1. 从深度为0的书开始,对每个叶节点枚举所有的可用特征
  2. 针对每个特征,把属于该节点的训练样本的该特征值升序排列,通过线性扫描的方式来决定该特征的最佳分裂点,并采用最佳分裂点时的收益
  3. 选择收益最大的特征作为分裂特征,用该特征的最佳分裂点作为分裂位置,把该节点生成出左右两个新的叶子结点,并为每个新节点关联新的样本集
  4. 回到第一步,继续递归直到满足特定条件。

那么如何计算上述收益呢?因为我们在某个节点二分成两个节点,分别是左L右R。因为除了当前待处理的节点,其他节点对应的中的值都不变,所以我们只需要考虑当前节点的值即可。

分类前的针对于该子节点的最优目标函数就是,分裂后则变成了,那么对于该目标函数来说,分裂后的收益为:

于是便可以用上式来决定最优分裂特征和最优特征分割点。

4.4 总结

所以xgboost的算法为:

  1. 算法在前向分步算法的每一步都新生成一颗决策树
  2. 拟合这棵树之前,计算损失函数在每个样本上的一阶导和二阶导,即
  3. 通过贪心策略生成一棵树,计算每个叶子结点的,计算预测值
  4. 把新生成的决策树加入,其中是学习率,为了抑制模型的过拟合

5. XGBoost的优化

XGBoost是DMLC开源在Github的Gradient Boosting框架,主要作者是陈天奇。它支持C++/Java/Python/R等语言,同样也支持Hadoop/Spark/Flink等分布式处理框架,且在数据竞赛中拥有优异的表现。

相比于普通的GBDT,XGBoost主要优点在于:
- 不仅支持决策树作为基分类器,还支持线性分类器
- 用到了函数的二阶泰勒展开,因此与损失函数更接近,收敛更快
- 在代价函数中加入了正则项,用于控制模型复杂度。正则项里包括了树的叶子节点个数和叶子结点输出值的L2范数,可以防止模型过拟合。
- Shrinkage,就是前面所述的,主要用于削弱每科树的影响,让后面有更大的学习空间。实际应用中,一般把设小点,迭代次数设大点。
- 列抽样(column sampling)。xgboost从随机森林算法中借鉴来的,支持列抽样可以降低过拟合,并且减少计算。
- 支持对缺失值的处理。对于特征值缺失的样本,xgboost可以学习这些缺失值的分裂方向。
- 支持并行。在对每颗树进行节点分裂时,需要计算每个特征的增益,选择最大的那个特征作为分裂特征,各个特征的增益计算可以开多线程进行
- 近似算法(Approximate Algorithm),树节点在分裂时,需要枚举每个可能的分割点。当数据没法一次性载入内存时,这种方法会很慢。xgboost提出了一种近似的方法去高效的生成候选分割点。

接下来选取几个比较重要的点详细讲讲

5.1 Approximate Split Finding Algorithm

树学习的关键问题就是找到最优分割点。常见的方法是枚举所有可能的分割点,称之为 exact greedy algorithm。

4

exact greedy algorithm计算量过大,而且当数据量较大没法全填入内存时,会很慢。因此xgboost引入了 approximate算法。

该算法对于某个特征,首先通过特征分布来确定若干值域分界点。然后根据这些值域分界点把样本分入桶中,对每个桶内的样本统计值进行累加,记为分界点的统计量。最后在分界点集合上进行贪心查找,得到的结果就是最佳分裂点的近似。

5

那么该如何寻找值域分界点呢?XGBoost中介绍了一种方法,叫加权分位数略图 Weighted Quantile Sketch

为了尽可能的逼近最佳分裂点,我们需要保证采样后的数据分布和原始数据分布尽可能一直。令表示每个训练样本的第维特征的值和对应的二阶导数。然后定义排序函数为

为什么使用二阶导h作为权重,参见Reference 6的附录部分。从公式上来看,当越大时,就越密集。

然后找到一组点满足:

其中,是采样率,因为,所以我们会得到个分界点。

6
(该图未考虑权重

5.2 Sparsity-aware Split Finding

实际运用中,输入很可能是稀疏的。有很多种可能造成数据是稀疏的:1) 数据中的缺失值; 2)较多的0值;3) 类似one-hot的特征工程。

对于这些缺失值,xgboost将样本分类到默认分支上去,而默认分支是由non-missing value学习得到的。具体算法如下:

7

由上可见,该算法只考虑non-missing entries ,因此计算复杂度是数据中的非缺失值个数的线性比值,对于稀疏数据来说,计算的很快。

对于缺失值的样本,算法分别将它们放到左节点和右节点,选取增益最大的一侧作为默认分类。

5.3 Column Block

XGBoost存储训练数据的数据结构被称为Block,每个Block内的数据采用CSC格式(Compressed Sparse Column)存储。

什么是CSC格式?CSC格式使用3个数组来存储稀疏矩阵。3个数组的定义如下:

举例如下:

存储的3个矩阵分别是:

实际上xgboost存储的时候,每一列都按照该列特征的取值排序好。这样一来的话,在寻找最优分割点,遍历所有特征的取值时,我们只需要遍历即可(可以多线程并行加速遍历不同列的特征)。

5.4 分布式实现

多机实现中,每个worker节点都会启动一个XGBoost进行,进程之间互相通信的数据包括:树模型的最新参数;每次分裂叶子节点时,为了计算最优split point,所需要从各个节点汇集的统计量,包括近似算法中的bucket信息等。各节点先将自身计算得到的信息传给Rank0节点,Rank0节点汇总后把树模型最新参数发给各个节点。

Reference

  1. 《统计学习方法》 李航
  2. 机器学习-一文理解GBDT的原理-20171001
  3. GBDT算法原理深入解析
  4. Gradient Boosting Decision Tree
  5. 机器学习算法中GBDT和XGBOOST的区别有哪些?
  6. Chen, Tianqi, and Carlos Guestrin. "Xgboost: A scalable tree boosting system." Proceedings of the 22nd acm sigkdd international conference on knowledge discovery and data mining. ACM, 2016.
  7. XGboost: A Scalable Tree Boosting System论文及源码导读
  8. Why xgboost is so fast?-Yafei Zhang

[1] 详细请参照《统计学习方法》第五章
[2] 为什么树模型是最好的
[3] 为什么系数默认为1?
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注