朴素贝叶斯算法仍然是流行的十大挖掘算法之一,该算法是有监督的学习算法,解决的是分类问题,如客户是否流失、是否值得投资、信用等级评定等多分类问题。**该算法的优点在于简单易懂、学习效率高、在某些领域的分类问题中能够与决策树、神经网络相媲美。**但由于该算法以自变量之间的独立(条件特征独立)性和连续变量的正态性假设为前提,就会导致算法精度在某种程度上受影响。接下来我们就详细介绍该算法的知识点及实际应用。
朴素贝叶斯分类器是一种基于贝叶斯定理的弱分类器,所有朴素贝叶斯分类器都假定样本每个特征与其他特征都不相关。举个例子,如果一种水果其具有红,圆,直径大概3英寸等特征,该水果可以被判定为是苹果。尽管这些特征相互依赖或者有些特征由其他特征决定,然而朴素贝叶斯分类器认为这些属性在判定该水果是否为苹果的概率分布上独立的。
朴素贝叶斯分类器很容易建立,特别适合用于大型数据集,众所周知,这是一种胜过许多复杂算法的高效分类方法。
尽管实际上独立假设常常是不准确的,但朴素贝叶斯分类器的若干特性让其在实践中能够取得令人惊奇的效果。特别地,各类条件特征之间的解耦意味着每个特征的分布都可以独立地被当做一维分布来估计。这样减轻了由于维度灾难带来的阻碍,当样本的特征个数增加时就不需要使样本规模呈指数增长。然而朴素贝叶斯在大多数情况下不能对类概率做出非常准确的估计,但在许多应用中这一点并不要求。
朴素贝叶斯分类器通过假设每一个特征的出现都是独立事件,简化了计算复杂度,也避免了样本稀疏的问题。虽然这一假设常常是不准确的,但朴素贝叶斯在实际工程中出乎意料地好用。因为很多应用并不在乎精确的类概率,只关心最后的分类结果。 然而,当特征之间相关性比较强,而我们又要求比较精确的类概率的时候,朴素贝叶斯就不够用了,即如果假设特征之间存在着概率依存关系,模型就变成了贝叶斯网络。贝叶斯网络是一种基于贝叶斯理论以DAG形式描述全局概率分布的一种统计方法,不属于分类器的一种,主要用于贝叶斯推断。
朴素贝叶斯(Naive Bayes)算法是机器学习中常见的基本算法之一,它主要被用来做分类任务。其理论基础是基于贝叶斯定理和条件独立性假设的一种分类方法。对于给定的训练数据集: $$ T={(x_1,y_1),(x_2,y_2),...,(x_n,y_n)} $$ 首先基于特征条件独立性假设学习联合概率分布$P(X,Y)$,然后基于此模型,对于任意的输入$x$,利用贝叶斯定理求出后验概率最大的$P(Y|X=x)$对应的$y$的取值。
基于以上的解释,我们知道:(1)该算法的理论核心是贝叶斯定理;(2)它是基于条件独立性假设这个强假设基础之上的,这也是它为什么被称为“朴素”的主要原因。
根据贝叶斯定理,对一个分类问题,给定样本特征$x$,样本属于类别$y$的概率是:
$$
P(Y|X)=\frac{P(X|Y)P(Y)}{P(X)}
$$
公式中的$X$是特征向量的维度,假设为$n$。因此,有:
$$
P(Y|X)=\frac{P(X^{(1)},X^{(2)},...,X^{(n)}|Y)P(Y)}{P(X)}
$$
我们以挑西瓜为例子,$Y$有两种取值“好瓜
”和"不是好瓜
"。我们先来规定一下符号,
由上式可知,朴素贝叶斯法的学习过程主要是估计$P(y=c_k)$以及$P(x^{(j)}|y=c_k)$。对于类别概率估计可以使用如下的公式进行估计: $$ P(Y=c_k)=\frac{\sum_{i=1}^N\ I(y_i=c_k)}{N},\ k=1,2,...,K $$ 但是对于条件概率$P(x^{(j)}|y=c_k)$的概率分布的估计,我们需要根据特征向量的每一维$X^{(i)}$分两种情况进行讨论:
- 若$X^{(i)}$为离散型变量;
- 若$X^{(i)}$为连续型变量
特征值为离散型变量时的参数估计
当特征值为离散型变量时,这时候参数估计非常简单,我们只需要计算出每个特征值在相应的类中出现的概率就可以了。直接使用如下的公式进行估计: $$ P(X^{(j)}=x^{(j)}|Y=c_k)=\frac{\sum_{i=1}^N\ I(X^{(j)}=x_i^{(j)},Y=c_k)}{\sum_{i=1}^{N}\ I(Y_i=c_k)} $$ 特征值为连续型变量时的参数估计
当特征值为离散型变量的时候,我们使用上式对条件概率分布$P(X^{(j)}=x^{(j)}|Y=c_k)$进行估计。但是如果当特征值$x^{(j)}$为连续型变量,此时这种估计就会失效,此时我们就需要一种新的方式进行估计。常用的方法就是假设特征$x^{(j)}$在条件$Y=c_k$时服从某一分布,如常见的高斯分布、多项式分布以及伯努利分布等。
以高斯分布为例,假设条件概率分布$P(X^{(j)}=x^{(j)}|Y=c_k)$服从如下分布: $$ P(X^{(j)}=x^{(j)}|Y=c_k)=\frac{1}{\sqrt{2\pi}\sigma}\text{exp}\left(-\frac{(x_i-u_y)}{2\sigma^2}\right) $$ 其中,参数$u_y$和$\sigma_y$分别表示高斯分布的均值和方差,它们可以使用最大似然估计的方法进行计算得到。
到这里好像方法已经介绍完了,但是有一个小问题需要注意:在上述的公式中,如果从样本中算出的概率值为0该怎么办呢?下面介绍一种简单方法,给学习步骤中的两个概率计算公式,分子和分母都分别加上一个常数,就可以避免这个问题。更新过后的公式如下:
$$
P(Y=c_k)=\frac{\sum_{i=-1}^N\ I(y_i=c_k)+\lambda}{N+K\lambda},\ k=1,2,...,K
$$
K是类的个数,
$$
P(X^{(j)}=x^{(j)}|Y=c_k)=\frac{\sum_{i=1}^N\ I(X^{(j)}=x_i^{(j)},Y=c_k)+\lambda}{\sum_{i=1}^{N}\ I(Y_i=c_k)+L_j\lambda},\ k=1,2,...,K
$$
特别的,当上述估计中$\lambda=1$时,此时称为拉普拉斯平滑(Laplace smoothing)。
优点:
- 既简单又快速,预测表现良好;
- 如果变量独立这个条件成立,相比Logistic回归等其他分类方法,朴素贝叶斯分类器性能更优,且只需少量训练数据;
- 相较于数值变量,朴素贝叶斯分类器在多个分类变量的情况下表现更好。若是数值变量,需要正态分布假设。
缺点:
- 如果分类变量的类别(测试数据集)没有在训练数据集总被观察到,那这个模型会分配一个0(零)概率给它,同时也会无法进行预测。这通常被称为“零频率”。为了解决这个问题,我们可以使用平滑技术,拉普拉斯估计是其中最基础的技术。
- 朴素贝叶斯也被称为bad estimator,所以它的概率输出predict_prob不应被太认真对待。
- 朴素贝叶斯的另一个限制是独立预测的假设。在现实生活中,这几乎是不可能的,各变量间或多或少都会存在相互影响。
scikit-learn里有3种朴素贝叶斯的模型:
高斯模型:适用于多个类型变量,假设特征符合高斯分布。
多项式模型:用于离散计数。如一个句子中某个词语重复出现,我们视它们每个都是独立的,所以统计多次,概率指数上出现了次方。
伯努利模型:如果特征向量是二进制(即0和1),那这个模型是非常有用的。不同于多项式,伯努利把出现多次的词语视为只出现一次,更加简单方便。
你可以根据特定数据集选取上述3个模型中的合适模型。
本节实战中,我们仍然使用iris数据集,详细步骤如下:
# -*- coding: utf-8 -*-
from sklearn.naive_bayes import GaussianNB
from sklearn.model_selection import train_test_split
from sklearn import datasets
# 加载数据集
iris = datasets.load_iris()
X = iris.data
y = iris.target
print('Sample num:', len(y))
# 随机分配训练集和测试集,比例大小为7:3
X_train, X_test, y_train, y_test = train_test_split(X, y, test_size = 0.3, random_state = 42)
# 初始化模型
clf = GaussianNB()
# 训练模型
clf.fit(X_train, y_train)
# 在测试集上预测
ans = clf.predict(X_test)
# 计算准确率
cnt = 0
for i in range(len(y_test)):
if ans[i] - y_test[i] < 1e-1:
cnt += 1
print(ans[i], ' ', y_test[i])
print("Accuracy:", (cnt * 100.0 / len(y_test)), "%")
# Accuracy:, 97.77777777777777%
“认识朴素贝叶斯”一节就参考的这篇知乎回答。
"朴素贝叶斯的优缺点"和“用scikit-learn进行朴素贝叶斯分类”参考此知乎回答。
"理解朴素贝叶斯分类"参考此专栏文章。
"拉普拉斯平滑"参考此专栏文章。
===
举例说明了朴素贝叶斯的求解过程。