机器学习-树模型理论(GDBT,xgboost,lightBoost,随机森林)
tree based ensemble algorithms
- 原始的Boost算法是在算法开始的时候,为每个样本赋上一个权重值,初始的时候,每个样本都是同样的重要。在每一步的训练中,得到的模型,会给出每个数据点的估计对错,根据判断的对错,在每一步的训练之后,会增加分错样本的权重,减少分类正确的样本的权重,如果在后续的每一步训练中,如果继续被分错,那么就会被严重的关注,也就是获得了一个比较高的权重。经过N次迭代之后,将会得到N个简单的分类器(base learner),然后将他们组装起来(可以进行加权,或者进行投票),得到一个最终的模型。
主要介绍以下几种ensemble的分类器(tree based algorithms)
xgboost
- xgboost能自动利用cpu的多线程,而且适当改进了gradient boosting,加了剪枝,控制了模型的复杂程度
- 传统的GBDT算法以CART作为基分类器,xgboost还可以支持线性分类器,相当于带L1和L2的逻辑斯谛回归或者线性回归
- 传统的GBDT在优化的时候,使用的是一阶导数信息,xgboost则对代价函数进行了二阶泰勒展开,同时用到了一阶导数和二阶导数。顺便提一下,xgboost工具支持自定义代价函数,只要函数可一阶和二阶求导。
- xgboost在代价函数中加入了正则项,用于控制模型的复杂度。正则项里面包括树的叶子节点的个数、每个叶子节点上输出的score的L2模的平方和。( 从Bias-variance tradeoff 的角度来说,正则化降低了模型的variance,使得到的模型更加简单,防止过拟合,这是xgboost优于传统的GBDT的一个特征)
- Shrinkage(缩减),相当于学习速率(xgboost中的eta)。xgboost在进行完一次迭代后,会将叶子节点的权重乘上该系数,主要是为了削弱每棵树的影响,让后面有更大的学习空间。实际应用中,一般把eta设置得小一点,然后迭代次数设置得大一点。(sklearn中的GBDT的实现也有学习速率)
- 列抽样(column sampling),借鉴了随机森林的做法,支持列抽样可以降低过拟合,同时减少了计算量,这也是xgboost异于传统gbdt的一个特性,sklearn中已经实现行采样和列采样,同样xgboost也是可以实现的。
- 对缺失值的处理。对于样本有特征确实的情况下,xgboost可以自动学习它的分裂方向。
- xgboost工具支持并行。xgboost的并行并不是tree粒度的并行,xgboost也是需要一次迭代完成之后,才能进行下一次迭代的。(第t次迭代的代价函数包含了前面t-1次的预测值)。xgboost的并行是特征粒度上的。决策树的学习最耗时的步骤是就是对特征进行排序(因为要确定最佳的分割点),xgboost在训练之前,预先对数据进行排序,然后保存成block结构,后面的迭代中重复的使用这个结构,大大的减少了计算量。这个结构也使并行成为可能。在进行节点分裂时,需要计算每个特征的信息增益,最终选择增益最大的那个特征去分裂,那么各个特征的增益计算就可以开多线程计算。
- 可并行的近似直方图算法。树节点在进行分裂时,需要计算每个特征的的每个分裂点的信息增益,即用贪心法枚举所有的可能的分割点。当数据无法一次性载入内存或者在分布式的情况下,贪心的算法效率就会变得很低,所以xgboost还提出了一种,可并行的近似直方图算法,用于高效的生成候选的分割点。
- xgboost的目标函数
- xgboost 分裂节点时所采用的公式
- 这个公式形式上与ID3算法与CART算法是一致的,两者做差,得到某种增益。为了限制树的生长,我们可以加入阈值,当增益大于阈值时才让节点分裂,上式中的gamma即为阈值,它是正则项里叶子节点数T的系数,所以xhboost在优化目标函数的同时相当于也做了预剪枝。
公式中的系数lambda,是正则项leaf score的L2模平方的系数,对leaf score做了平滑,也起到了防止过拟合的作用,这还是传统GBDT里不具备的特性。
- 总结下来就是两者的区别:
- xgboost里面的基学习器除了用tree(gbtree),也可用线性分类器(gblinear)。而GBDT则特指梯度提升决策树算法。
- xgboost相对于普通gbm的实现,可能具有以下的一些优势:
- 显式地将树模型的复杂度作为正则项加在优化目标
- 公式推导里用到了二阶导数信息,而普通的GBDT只用到一阶
- 允许使用column(feature) sampling来防止过拟合,借鉴了Random Forest的思想。(sklearn里的gbm好像也有类似实现)
- 实现了一种分裂节点寻找的近似算法,用于加速和减小内存消耗。
- 节点分裂算法能自动利用特征的稀疏性。
- data事先排好序并以block的形式存储,利于并行计算
- penalty function Omega主要是对树的叶子数和叶子分数做惩罚,这点确保了树的简单性
- 支持分布式计算可以运行在MPI,YARN上,得益于底层支持容错的分布式通信框架rabit。
- The correct answer is marked in red. Please consider if this visually seems a reasonable fit to you. The general principle is we want both a simple and predictive model. The tradeoff between the two is also referred as bias-variance tradeoff in machine learning.
lightGBM: 基于决策树算法的分布式梯度提升框架
- lightGBM 与xgboost的区别:
- xgboost使用的是pre-sorted算法(对所有的特征都按照特征的数值进行预排序,在遍历分割点的时候用O(data)的代价函数找个一个特征的最好分割点,能够更加精确的找到数据的分割点。
- lightGBM 使用的是histogram算法,占用内存更低,数据分割的复杂度更低。
- 决策树生长策略上
- xgboost采用的是level-wise生长策略,能够同时分类同一层的叶子,从而进行多线程优化,不容易过拟合,但是不加区分的对待同一层的叶子,带来了很多没有必要的开销(有很多的叶子分裂增益较低,没有必要进行搜索和分裂)
- lightGBM采用的是leaf-wise的生长策略,每次从当前的叶子中找到分裂增益最大的(一般也是数据量最大)的一个叶子进行分裂,如此循环;但是生长出的决策树枝叶过多,产生过拟合,lightGBM在leaf-wise上增加了一个最大深度的限制,在保证高效率的同时防止过拟合。
- 另一个巧妙的优化是histogram做差加速,一个容易观察到的现象:一个叶子的直方图可以由它的父节点的直方图与它兄弟的直方图做差得到。
GBDT(Gradient Boosting Decison Tree)
- GBDT中使用的都是回归树,GBDT用来做回归预测,调整后也可以用于分类,设定阈值,大于阈值为正例,反之为负例,可以发现多种有区分性的特征以及特征组合。
- GBDT是把所有树的结论累加起来做最终结论,GBDT的核心就在于,每一棵树学的是之前所有树结论和的残差,这个残差就是把一个加预测值后能得到真实值的累加量。
- 比如A的真实年龄是18岁,但第一棵树的预测年龄是12岁,差了6岁,即残差为6岁。那么在第二棵树里我们把A的年龄设为6岁去学习,如果第二棵树真的能把A分到6岁的叶子节点,那累加两棵树的结论就是A的真实年龄;如果第二棵树的结论是5岁,则A仍然存在1岁的残差,第三棵树里A的年龄就变成1岁,继续学。 Boosting的最大好处在于,每一步的残差计算其实变相地增大了分错instance的权重,而已经分对的instance则都趋向于0。这样后面的树就能越来越专注那些前面被分错的instance。
用公式来表示提升树的部分原理
GBDT划分标准默认是friedman_mse可以查看sklearn 官方文档中GBDT的参数说明
- Gradient Boost与传统的Boost的区别是
- 每一次的计算是为了减少上一次的残差(residual),而为了消除残差,我们可以在残差减少的梯度(Gradient)方向上建立一个新的模型。
- 所以说,在Gradient Boost中,每个新的模型的建立是为了使得之前模型的残差往梯度方向减少。
- Shrinkage(缩减)的思想认为,每次走一小步逐渐逼近结果的效果,要比每次迈一大步很快逼近结果的方式更容易避免过拟合。
- 即它不完全相信每一棵残差树,他认为每棵树只学到了真理的一部分,累加的时候只累加一小部分,每次通过多学几棵树弥补不足。
- 本质上,Shrinkage为每棵树设置了一个weight,累加时要乘以这个weight,但和Gradient并没有关系。
The advantages of GBRT are:
- Natural handling of data of mixed type (= heterogeneous features)
- 可以处理不同性质的属性,数值特征与category特征,
- 数值特征需要进行数据的预处理
- Predictive power
Robustness to outliers in output space (via robust loss functions)
The disadvantages of GBRT are:
- Scalability, due to the sequential nature of boosting it can hardly be parallelized.
- Boost是一个串行过程,不好并行化,而且计算复杂度高,同时不太适合高维稀疏特征。
随机森林
- 是随机的方式建立一个森林,森林里面有很多的决策树组成,随机森林的每一决策树质检是没有关联的。在得到随机森林之后,当有一个新的样本输进的时候,就让森林中的每一棵决策树进行判断,判断样本属于哪一类,然后看哪一类被选择最多,就预测这个样本为这一类。
- 随机采样
随机行采样,采用有放回的方式,也就是在采样得到的样本集合中,可能有重复的样本。
假如输入的样本为N个,那么采样的样本也是N个。这使得在训练的时候,每棵树的输入的样本都不是全部的样本,使得相对不容易出现over-fitting。- 随机列采样,从M个feature中,选择m个(m<<M)。
对采样之后的数据使用完全分裂的方式建立决策树,这样的决策树的某个叶子节点要么无法继续分裂,要么里面所有的样本都是指向的同一分类。
- 随机森林 中同样有剪枝,限制决策树的最大深度,以及最小的样本分裂,最小的节点样本数目,样本分裂节点的信息增益或gini系数必须达到的阈值
- 随机森林用于分类的话,划分标准是entropy或者gini系数
随机森林用于回归的话,划分标准是mse(mean squared error)或者mae(mean absolute error)
Why is it called random then?
- 假如我们的数据有1000行,30列,随机森林的算法中,有两个不同水平的随机
- At row level
- 每棵树随机抽取了样本 数据的一部分,每棵树都是独立训练的,给出的预测结果也是独立的
- At column level, feature random selection at node level of the decision tree
- 每棵树在分支的时候,只使用了部分特征进行熵或者基尼系数的计算,比如说我们选择了3个特征进行分支,树的第一个分支的时候使用了第1,2,4个特征,然后计算这三个特征的指标(比如说Gini coefficients或者其他的一些指标来选择最佳的节点),这三个特征中某个特征用来分支,第二个分支的时候,重复就行这个操作,这个时候的特征就可能是第7,9,10个特征,然后重新计算特征选择的指标,然后寻找其中的最佳的特征进行分支
参考链接medium给出的解释,random forest的其中的作者指出的错误https://medium.com/theboredhuman/random-forests-explained-intuitively-2cecb9e1a7b5
- 每棵树在分支的时候,只使用了部分特征进行熵或者基尼系数的计算,比如说我们选择了3个特征进行分支,树的第一个分支的时候使用了第1,2,4个特征,然后计算这三个特征的指标(比如说Gini coefficients或者其他的一些指标来选择最佳的节点),这三个特征中某个特征用来分支,第二个分支的时候,重复就行这个操作,这个时候的特征就可能是第7,9,10个特征,然后重新计算特征选择的指标,然后寻找其中的最佳的特征进行分支
决策树
- ID3 信息增益:熵(数据的不确定性程度)的减少;一个属性的信息增益量越大,这个属性作为一棵树的根节点就能使这棵树更简洁。
信息增益=分裂前的熵 – 分裂后的熵
面对类别较少的离散数据时效果较好,但如果面对连续的数据(如体重、身高、年龄、距离等),或者每列数据没有明显的类别之分(最极端的例子的该列所有数据都独一无二),即每个值对应一类样本 C4.5信息增益比:克服了ID3用信息增益选择属性时偏向选择取值多的属性的不足(某个属性存在大量的不同值,在划分时将每个值分为一个结点)
- CART 使用基尼系数进行分类
基尼指数Gini(D)表示集合D的不确定性,基尼指数Gini(D,A)表示经A=a分割后集合D的不确定性。基尼指数值越大,样本集合的不确定性也就越大,这一点与熵相似。
- 分类与回归树(CART):二叉树形式,分类时:根据Gini指数选择划分特征
回归时:Los为 平方损失函数,最小化均方误差选择划分特征,切分点(值)将数据切分成两部分,用平方误差最小的准则(最小二乘法)求解每个单元上的最优输出值(每个叶子节点上的预测值为所有样本的平均值)。
用选定的对(j,s)划分区域并决定相应的输出值,每个叶子节点上的预测值为所有样本的平均值:
决策树的生成通常使用 信息增益最大、信息增益比最大或基尼指数最小作为特征选择的准则。
[参考]:https://medium.com/theboredhuman/random-forests-explained-intuitively-2cecb9e1a7b5
机器学习-树模型理论(GDBT,xgboost,lightBoost,随机森林)的更多相关文章
- web安全之机器学习入门——3.2 决策树与随机森林
目录 简介 决策树简单用法 决策树检测P0P3爆破 决策树检测FTP爆破 随机森林检测FTP爆破 简介 决策树和随机森林算法是最常见的分类算法: 决策树,判断的逻辑很多时候和人的思维非常接近. 随机森 ...
- Python机器学习笔记——随机森林算法
随机森林算法的理论知识 随机森林是一种有监督学习算法,是以决策树为基学习器的集成学习算法.随机森林非常简单,易于实现,计算开销也很小,但是它在分类和回归上表现出非常惊人的性能,因此,随机森林被誉为“代 ...
- 机器学习(六)—随机森林Random Forest
1.什么是随机采样? Bagging可以简单的理解为:放回抽样,多数表决(分类)或简单平均(回归): Bagging的弱学习器之间没有boosting那样的联系,不存在强依赖关系,基学习器之间属于并列 ...
- R语言︱决策树族——随机森林算法
每每以为攀得众山小,可.每每又切实来到起点,大牛们,缓缓脚步来俺笔记葩分享一下吧,please~ --------------------------- 笔者寄语:有一篇<有监督学习选择深度学习 ...
- Python中随机森林的实现与解释
使用像Scikit-Learn这样的库,现在很容易在Python中实现数百种机器学习算法.这很容易,我们通常不需要任何关于模型如何工作的潜在知识来使用它.虽然不需要了解所有细节,但了解机器学习模型是如 ...
- R语言之Random Forest随机森林
什么是随机森林? 随机森林就是通过集成学习的思想将多棵树集成的一种算法,它的基本单元是决策树,而它的本质属于机器学习的一大分支——集成学习(Ensemble Learning)方法.随机森林的名称中有 ...
- 用Python实现随机森林算法,深度学习
用Python实现随机森林算法,深度学习 拥有高方差使得决策树(secision tress)在处理特定训练数据集时其结果显得相对脆弱.bagging(bootstrap aggregating 的缩 ...
- chapter02 三种决策树模型:单一决策树、随机森林、GBDT(梯度提升决策树) 预测泰坦尼克号乘客生还情况
单一标准的决策树:会根每维特征对预测结果的影响程度进行排序,进而决定不同特征从上至下构建分类节点的顺序.Random Forest Classifier:使用相同的训练样本同时搭建多个独立的分类模型, ...
- OpenCV:使用 随机森林与GBDT
随机森林顾名思义,是用随机的方式建立一个森林.简单来说,随机森林就是由多棵CART(Classification And Regression Tree)构成的.对于每棵树,它们使用的训练集是从总的训 ...
随机推荐
- 使用JfreeChart生成图表遇到的问题
生成的图片不显示 需要在web.xml中配置一个指定的Servlet <servlet> <servlet-name>DisplayChart</servlet-name ...
- PMM 对MYSQL 的监控配制
系统选择: centos 7.2 关闭防火墙: systemctl stop firewalld.service systemctl disable firewalld.s ...
- 【独家】硅谷创业公司在中国常跌的五个坑|禾赛科技CEO李一帆柏林亚太周主题演讲
[独家]硅谷创业公司在中国常跌的五个坑|禾赛科技CEO李一帆柏林亚太周主题演讲 李一帆 Xtecher特稿作者 关注 Xtecher推荐 演讲者:李一帆 翻译:晓娜 网址:www.xt ...
- arcgis 获得工具箱工具的个数
import arcgisscripting import string; gp = arcgisscripting.create(9.3); ##多少个工具箱 toolboxes = gp.list ...
- Ubuntu 18.04 安装Virtual Box or VMWare workstation Pro 14
Linux相关的知识:https://www.cnblogs.com/dunitian/p/4822808.html#linux Virtual Box:sudo apt-get install vi ...
- SpringCloud无废话入门01:最简SpringCloud应用
1.创建Parent Parent很简单,创建一个空的maven项目,pom如下: <?xml version="1.0" encoding="UTF-8" ...
- java 根据系统日期获取前一天、后一天时间(根据初始日期推算出期望(向前/向后)日期)
1.情景展示 java 根据系统当前日期获取前一天日期.后一天日期,或者根据初始日期推算出期望(向前/向后)日期. 2.解决方案 导包 import java.text.ParseExcepti ...
- github远程建了分支,本地看不到的问题
原因:Git branch -a 查看的是本地仓库的所有分支 远程新建的没有同步前 就是看不到 解决:$git checkout master //首先切到master分支 $git pull ...
- Charles配置问题
1. 手机访问chls.pro/ssl下载证书时候,用常用安卓手机不同的浏览器(可以多试几种浏览器) 会出现两种情况,一种是直接打开下载getssl.crt文件 一种是没有反应,直接打开网页了 这时候 ...
- Android 异常 android.os.NetworkOnMainThreadException
近期在实现一个Android下的数据採集的SDK,收集用户使用数据使用HTTP发送到云平台.进行数据分析.但在发送数据时报例如以下错误: Caused by: android.os.NetworkOn ...