Boosting Model: AdaBoost GBDT和XGBoost
1. AdaBoost
AdaBoost是借助Boosting思想解决二分类问题。其模型的基本思路是通过学习多个分类器(比如单层的决策树Decision Stump),每一轮学习新的分类器,更加关注上一轮被错误划分的样本点。因此该模型需要重点关注两个点:1.如何确定分类器的权重;2.如何调整样本权重,从而在下一轮中更加“关注”错分的样本。
介绍Adaboost可以从不同的角度来入手,比如通过列出单回合的指数损失函数,令其值最小化,可以得到新一轮的分类器如何设置权重;通过列出additive model的目标函数使用泰勒展开,可以推导出样本权重的调整规则。但下文将直接使用AdaBoost算法规则的Intuition来解释分类的权重、样本的权重为什么这么设置。
首先给出分类错误率的定义,对于第轮,指示函数,样本权重:
其实可以证明最小化指数损失函数等价于最小化错误率。
在生成一个分类器后,我们要生成该分类器的权重。理所应当的想法是该权重和分类器的错误率相关:当错误率为时,分类器相当于乱猜,权重应该为0;如果分类器的错误率为0,则权重应该非常大。则权重可以设置为:
再考虑样本权重设置,我们希望本轮的分类器在处理下一轮调整后的样本时,表现地非常差:
因此在轮,我们应该将上一轮错误划分样本和正确划分的样本权重调整为相同。一种最直接的调整方式就是乘上对方,并除以一个固定值(两者之和),即可以变形为:分别乘以,在做缩放(Scale),可以得到:
这里最后再提一个概念,即样本权重调整(Weight Resampling),在具体操作时不是将权重直接乘以,而是。对于无法进行该操作的分类器,可以采用Resampling的方法,即通过调整抽样概率实现样本权重的调整。
2. GBDT和XGBoost
上面介绍了Adaboost,而Adaboost实际上可以用最速下降法(或者我们下面要讲的Gradient Boosting)来解释,其实质是一个使用指数损失函数解决二分类问题的特殊情况下的最速下降模型。下文要考虑的是,如果我们的损失函数不再是使用指数损失函数,更一般情况下的“Adaboost”便是梯度提升模型。
2.1 回归树
在讲GDBT(Gradient Boosting Decision Tree)之前, 我们先讲回归树。
不同于分类树算法使用熵作为分枝的条件,回归树采用最小二乘误差来确定分支的条件,具体操作是:
- 挑选待分割的变量(依次遍历各待分割变量,找到最好的那个),
- 固定变量x后,遍历地设置切分点s,划分为和两个子节点,子节点的输出值是各自的均值,然后计算两个子节点各自的二乘误差并求和。挑选使二乘误差最小的切分点。对子节点递归上述操作,指导满足最终停止条件。
2.2 提升树
Boosting 意为提升;其在操作时,先从初始训练集训练出一个基学习器;再根据学习器的结果对训练样本的分布进行调整,使之前预测错误的样本在后续的预测中得到更多的关注,通常预测准确性高的学习器具有较高的学习的权重。
Boosting 使用公式来表达:
提升树(Boosting Tree)则是在每轮生成树的过程中使用前面上一颗树各样本的残差项作为预测值(),去生成新的树。
2.3 梯度提升树
有些时候我们通过谨慎地区分各个概念可以方便我们理解各个模型的关联关系,这里我们要需要谨慎对待的概念是GB算法(Gradient Boosting Algorithm)和GBDT(Gradient Boosting Decision Tree, 又名Multivariate Adaptive Regreesion Tree或MART),后者只是前者的一种具体实现。下面首先要介绍Gradient Boosting算法:一种通过确定任意损失函数的梯度,以及更新步长来“boosting”的算法;接下来以GBDT为例,介绍Gradient Boosting算法的一种通过回归树作为弱学习器的实现。
2.3.1 Gradient Boosting
梯度提升算法是使用函数梯度下降(Function Gradient Descent)的方法,使用连续多个函数(分类器)沿着函数梯度下降的方向做优化,从而使损失函数最小化。理论上任意可以微的损失函数都适用于该算法。下面是算法实现的基本步骤:
- 初始化常数项作为初始函数,该参数里可以理解为线性模型的常数项bias
- 对于迭代轮次m=1到M:
- 计算梯度残差(gradient residual):
- 适用学习器拟合该残差, 训练集为
- 计算步长
- 适用更新方程:
- 得到最终模型:
2.3.2 Gradient Boosting Decision Tree
而GBDT是将上述GB算法中的学习器替换为回归树,并对固定步长做了修正,令每个叶子结点的相互独立,因此有,其中为叶子结点的序号。
因此GBDT与一般形式的GB算法相比,只在一维最优化步长处有差异,下面给出步长的方程式:
2.4 XGBoost
模型细节上的变化,XGBoost是GBDT的基础上,做了一些细节上的修改。这些修改包括:1.加入正则项(包括对叶子结点数目的一次项惩罚和回归树输出值的二次项惩罚), 2.将泰勒一阶展开修改修改为二阶展开。3. 两个实践操作被加入到模型中防止过拟合: (1)Shrinkage: 给每一步的叶子结点的值乘上一个小于1的值,类似于随机梯度下降中调整学习速率,给未来的子树更多的提升空间。(2)column sampling, 类似于随机森林里的特征采样。
Xgboost在每一颗子树时,当前阶段的损失函数可以由一阶梯度、二阶梯度的值(每个样本有两个值)以及各叶子的值(变量)表示,因此可以列出损失函数是叶子结点值的最优化问题,不同的特征分割方法决定了叶子值的大小,以及损失函数的最优值。因此每一次分割时,可以计算当前的损失函数最优值是多少,作为是否进行分割的依据。下式中分别为一阶、二阶梯度,为叶子结点的数目, 为树的结构。
树的训练不是最优的,而是基于贪心算法的,每一次分割会增加一个左、右子树,带入上述损失函数的计算公式可以得到一个loss函数值的增加,同时上一个结点消失带来的损失值的消失要减掉。
除此之外,Xgboost还有一些工程上的优化
Split Finding Algorithm, 传统做法是对所有样本在该特征上进行排序,然后遍历所有可能的分割点,不同的特征在该步骤是可以同时进行的,但该前提是数据能够装入内存。根据该问题Xgboost的做法是:通过计算给特征的百分位数,(或者Weighted Quantile,其中权重为样本的二阶导数),将该特征上的样本分成几个桶,计算各个桶中的一、二阶梯度,然后计算以各split点下的loss值如何。或者直接构造一、二阶梯度的histograms。这几种划分的方法,Xgboost都是支持的。
Sparsity-aware Split Finding,数据存在missing value,0值,one-hot编码等常常会导致特征是稀疏的。比如对于missing value,决策树会将missing值分别尝试都放在左子树或右子树,然后看具体哪个更好。
Cache-aware Access提高CPU cache的利用率。由于数据的方式是按照某属性的sorted之后的顺序访问的,并不是对raw data在连续内存上的访问,对造成cache miss问题影响速度。对于exact greedy algorithm(不分桶),一种优化方法是是每个线程里申请一块buffer,把gradient信息放进去,在mini-batch level做梯度的accumulation。对于approximate algorithms(分桶),通过将数据的block size进行调整,来提高cpu cache的命中率,每个block样本太多导致cache miss, 每个block样本太小导致并行计算的效率下降。评估后每个block的样本数量为时,效果最好。
out-of-core computation减少block从磁盘读写花费的时间。使用的方法包括block列级别的压缩,block的分片(当数据存放在多个磁盘上时,分片的方法提高磁盘的读写)
3 Boosting算法为什么有效
- 间接达到Regularization的目的 是有多个学习器然后“整合”,而“整合”本身具有Regularization的性质。
4 Staking模型融合
Boosting的思想是通过若干个较弱的学习器通过组合的方法得到模型的最终的预测结果,而Stacking的思想是通过若干学习器(第一层:初级学习器M个)分别学习原始输入,比如一个样本依次输入三个初级学习器,分别得到,将其作为该样本下一层学习器(次级学习器)的输入,这样我们就把该样本数据重构为,其中作为次级模型的输入。
但是上述的做法有较大的可能会过拟合(随机性太少),则可以采用K折交叉检验(或者留一法)。比如k折交叉检验中,对于第一组样本数据,我们使用第2到K组数据分别作为训练集,训练初级学习器,然后依次对第一组样本的各个样本进行预测,得出预测值作为第一组样本中样本在次级学习机的输入。