CategoryResourceRepost/极客时间专栏/数据分析实战45讲/第二模块:数据分析算法篇/18丨决策树(中):CART,一棵是回归树,另一棵是分类树.md
louzefeng d3828a7aee mod
2024-07-11 05:50:32 +00:00

14 KiB
Raw Blame History

上节课我们讲了决策树基于信息度量的不同方式我们可以把决策树分为ID3算法、C4.5算法和CART算法。今天我来带你学习CART算法。CART算法英文全称叫做Classification And Regression Tree中文叫做分类回归树。ID3和C4.5算法可以生成二叉树或多叉树而CART只支持二叉树。同时CART决策树比较特殊既可以作分类树又可以作回归树。

那么你首先需要了解的是,什么是分类树,什么是回归树呢?

我用下面的训练数据举个例子,你能看到不同职业的人,他们的年龄不同,学习时间也不同。如果我构造了一棵决策树,想要基于数据判断这个人的职业身份,这个就属于分类树,因为是从几个分类中来做选择。如果是给定了数据,想要预测这个人的年龄,那就属于回归树。


分类树可以处理离散数据,也就是数据种类有限的数据,它输出的是样本的类别,而回归树可以对连续型的数值进行预测,也就是数据在某个区间内都有取值的可能,它输出的是一个数值。

CART分类树的工作流程

通过上一讲我们知道决策树的核心就是寻找纯净的划分因此引入了纯度的概念。在属性选择上我们是通过统计“不纯度”来做判断的ID3是基于信息增益做判断C4.5在ID3的基础上做了改进提出了信息增益率的概念。实际上CART分类树与C4.5算法类似,只是属性选择的指标采用的是基尼系数。

你可能在经济学中听过说基尼系数它是用来衡量一个国家收入差距的常用指标。当基尼系数大于0.4的时候说明财富差异悬殊。基尼系数在0.2-0.4之间说明分配合理,财富差距不大。

基尼系数本身反应了样本的不确定度。当基尼系数越小的时候说明样本之间的差异性小不确定程度低。分类的过程本身是一个不确定度降低的过程即纯度的提升过程。所以CART算法在构造分类树的时候会选择基尼系数最小的属性作为属性的划分。

我们接下来详解了解一下基尼系数。基尼系数不好懂,你最好跟着例子一起手动计算下。

假设t为节点那么该节点的GINI系数的计算公式为


这里p(Ck|t)表示节点t属于类别Ck的概率节点t的基尼系数为1减去各类别Ck概率平方和。

通过下面这个例子,我们计算一下两个集合的基尼系数分别为多少:

集合16个都去打篮球

集合23个去打篮球3个不去打篮球。

针对集合1所有人都去打篮球所以p(Ck|t)=1因此GINI(t)=1-1=0。

针对集合2有一半人去打篮球而另一半不去打篮球所以p(C1|t)=0.5p(C2|t)=0.5GINI(t)=1-0.50.5+0.50.5=0.5。

通过两个基尼系数你可以看出集合1的基尼系数最小也证明样本最稳定而集合2的样本不稳定性更大。

在CART算法中基于基尼系数对特征属性进行二元分裂假设属性A将节点D划分成了D1和D2如下图所示


节点D的基尼系数等于子节点D1和D2的归一化基尼系数之和用公式表示为


归一化基尼系数代表的是每个子节点的基尼系数乘以该节点占整体父亲节点D中的比例。

上面我们已经计算了集合D1和集合D2的GINI系数得到

所以在属性A的划分下节点D的基尼系数为

节点D被属性A划分后的基尼系数越大样本集合的不确定性越大也就是不纯度越高。

如何使用CART算法来创建分类树

通过上面的讲解你可以知道CART分类树实际上是基于基尼系数来做属性划分的。在Python的sklearn中如果我们想要创建CART分类树可以直接使用DecisionTreeClassifier这个类。创建这个类的时候默认情况下criterion这个参数等于gini也就是按照基尼系数来选择属性划分即默认采用的是CART分类树。

下面我们来用CART分类树给iris数据集构造一棵分类决策树。iris这个数据集我在Python可视化中讲到过实际上在sklearn中也自带了这个数据集。基于iris数据集构造CART分类树的代码如下

# encoding=utf-8
from sklearn.model_selection import train_test_split
from sklearn.metrics import accuracy_score
from sklearn.tree import DecisionTreeClassifier
from sklearn.datasets import load_iris
# 准备数据集
iris=load_iris()
# 获取特征集和分类标识
features = iris.data
labels = iris.target
# 随机抽取33%的数据作为测试集,其余为训练集
train_features, test_features, train_labels, test_labels = train_test_split(features, labels, test_size=0.33, random_state=0)
# 创建CART分类树
clf = DecisionTreeClassifier(criterion='gini')
# 拟合构造CART分类树
clf = clf.fit(train_features, train_labels)
# 用CART分类树做预测
test_predict = clf.predict(test_features)
# 预测结果与测试集结果作比对
score = accuracy_score(test_labels, test_predict)
print("CART分类树准确率 %.4lf" % score)

运行结果:

CART分类树准确率 0.9600

如果我们把决策树画出来,可以得到下面的图示:


首先train_test_split可以帮助我们把数据集抽取一部分作为测试集这样我们就可以得到训练集和测试集。

使用clf = DecisionTreeClassifier(criterion=gini)初始化一棵CART分类树。这样你就可以对CART分类树进行训练。

使用clf.fit(train_features, train_labels)函数将训练集的特征值和分类标识作为参数进行拟合得到CART分类树。

使用clf.predict(test_features)函数进行预测传入测试集的特征值可以得到测试结果test_predict。

最后使用accuracy_score(test_labels, test_predict)函数传入测试集的预测结果与实际的结果作为参数得到准确率score。

我们能看到sklearn帮我们做了CART分类树的使用封装使用起来还是很方便的。

CART回归树的工作流程

CART回归树划分数据集的过程和分类树的过程是一样的只是回归树得到的预测结果是连续值而且评判“不纯度”的指标不同。在CART分类树中采用的是基尼系数作为标准那么在CART回归树中如何评价“不纯度”呢实际上我们要根据样本的混乱程度也就是样本的离散程度来评价“不纯度”。

样本的离散程度具体的计算方式是先计算所有样本的均值然后计算每个样本值到均值的差值。我们假设x为样本的个体均值为u。为了统计样本的离散程度我们可以取差值的绝对值或者方差。

其中差值的绝对值为样本值减去样本均值的绝对值:


方差为每个样本值减去样本均值的平方和除以样本个数:


所以这两种节点划分的标准分别对应着两种目标函数最优化的标准即用最小绝对偏差LAD或者使用最小二乘偏差LSD。这两种方式都可以让我们找到节点划分的方法通常使用最小二乘偏差的情况更常见一些。

我们可以通过一个例子来看下如何创建一棵CART回归树来做预测。

如何使用CART回归树做预测

这里我们使用到sklearn自带的波士顿房价数据集该数据集给出了影响房价的一些指标比如犯罪率房产税等最后给出了房价。

根据这些指标我们使用CART回归树对波士顿房价进行预测代码如下

# encoding=utf-8
from sklearn.metrics import mean_squared_error
from sklearn.model_selection import train_test_split
from sklearn.datasets import load_boston
from sklearn.metrics import r2_score,mean_absolute_error,mean_squared_error
from sklearn.tree import DecisionTreeRegressor
# 准备数据集
boston=load_boston()
# 探索数据
print(boston.feature_names)
# 获取特征集和房价
features = boston.data
prices = boston.target
# 随机抽取33%的数据作为测试集,其余为训练集
train_features, test_features, train_price, test_price = train_test_split(features, prices, test_size=0.33)
# 创建CART回归树
dtr=DecisionTreeRegressor()
# 拟合构造CART回归树
dtr.fit(train_features, train_price)
# 预测测试集中的房价
predict_price = dtr.predict(test_features)
# 测试集的结果评价
print('回归树二乘偏差均值:', mean_squared_error(test_price, predict_price))
print('回归树绝对值偏差均值:', mean_absolute_error(test_price, predict_price)) 

运行结果(每次运行结果可能会有不同):

['CRIM' 'ZN' 'INDUS' 'CHAS' 'NOX' 'RM' 'AGE' 'DIS' 'RAD' 'TAX' 'PTRATIO' 'B' 'LSTAT']
回归树二乘偏差均值: 23.80784431137724
回归树绝对值偏差均值: 3.040119760479042

如果把回归树画出来,可以得到下面的图示(波士顿房价数据集的指标有些多,所以树比较大):

你可以在这里下载完整PDF文件。

我们来看下这个例子首先加载了波士顿房价数据集得到特征集和房价。然后通过train_test_split帮助我们把数据集抽取一部分作为测试集其余作为训练集。

使用dtr=DecisionTreeRegressor()初始化一棵CART回归树。

使用dtr.fit(train_features, train_price)函数将训练集的特征值和结果作为参数进行拟合得到CART回归树。

使用dtr.predict(test_features)函数进行预测传入测试集的特征值可以得到预测结果predict_price。

最后我们可以求得这棵回归树的二乘偏差均值,以及绝对值偏差均值。

我们能看到CART回归树的使用和分类树类似只是最后求得的预测值是个连续值。

CART决策树的剪枝

CART决策树的剪枝主要采用的是CCP方法它是一种后剪枝的方法英文全称叫做cost-complexity prune中文叫做代价复杂度。这种剪枝方式用到一个指标叫做节点的表面误差率增益值以此作为剪枝前后误差的定义。用公式表示则是


其中Tt代表以t为根节点的子树C(Tt)表示节点t的子树没被裁剪时子树Tt的误差C(t)表示节点t的子树被剪枝后节点t的误差|Tt|代子树Tt的叶子数剪枝后T的叶子数减少了|Tt|-1。

所以节点的表面误差率增益值等于节点t的子树被剪枝后的误差变化除以剪掉的叶子数量。

因为我们希望剪枝前后误差最小,所以我们要寻找的就是最小α值对应的节点,把它剪掉。这时候生成了第一个子树。重复上面的过程,继续剪枝,直到最后只剩下根节点,即为最后一个子树。

得到了剪枝后的子树集合后,我们需要用验证集对所有子树的误差计算一遍。可以通过计算每个子树的基尼指数或者平方误差,取误差最小的那个树,得到我们想要的结果。

总结

今天我给你讲了CART决策树它是一棵决策二叉树既可以做分类树也可以做回归树。你需要记住的是作为分类树CART采用基尼系数作为节点划分的依据得到的是离散的结果也就是分类结果作为回归树CART可以采用最小绝对偏差LAD或者最小二乘偏差LSD作为节点划分的依据得到的是连续值即回归预测结果。

最后我们来整理下三种决策树之间在属性选择标准上的差异:

  • ID3算法基于信息增益做判断
  • C4.5算法,基于信息增益率做判断;
  • CART算法分类树是基于基尼系数做判断。回归树是基于偏差做判断。
  • 实际上这三个指标也是计算“不纯度”的三种计算方式。

    在工具使用上我们可以使用sklearn中的DecisionTreeClassifier创建CART分类树通过DecisionTreeRegressor创建CART回归树。

    你可以用代码自己跑一遍我在文稿中举到的例子。


    最后给你留两道思考题吧你能说下ID3C4.5以及CART分类树在做节点划分时的区别吗第二个问题是sklearn中有个手写数字数据集调用的方法是load_digits()你能否创建一个CART分类树对手写数字数据集做分类另外选取一部分测试集统计下分类树的准确率

    欢迎你在评论下面留言,与我分享你的答案。也欢迎点击“请朋友读”,把这篇文章分享给你的朋友或者同事,一起交流。