数学建模社区-数学中国

标题: 朴素贝叶斯分类器_以python为工具【Python机器学习系列(十三)】 [打印本页]

作者: 杨利霞    时间: 2022-9-5 16:52
标题: 朴素贝叶斯分类器_以python为工具【Python机器学习系列(十三)】
朴素贝叶斯分类器_以python为工具【Python机器学习系列(十三)】
2 H# `/ B# o% z! I' @4 [
' G- l" ~  P& {- Z$ K" y! r文章目录6 W4 }$ ?; z+ w( R3 q4 |
1. 朴素贝叶斯算法原理
9 V9 |, ?* ^9 H2. sklearn提供的朴素贝叶斯算法5 N% @2 b; h. w; U" h  M. u
3. 伯努利朴素贝叶斯 BernoulliNB()
/ r- S2 ?& F& ]. d4. 多项式朴素贝叶斯 MultinomialNB()) K3 `, `+ y- X! \% z6 r
5. 高斯朴素贝叶斯 GaussianNB()
3 R+ _3 N- R: s, B- T  R3 S      ʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞ
* o/ ]; H, |. b' T( B7 i
9 _5 [6 b" j: ?: w, O) r/ e' |# J    ʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞʚʕ̯•͡˔•̯᷅ʔɞ
& W- p; l: q) s6 n: v$ p( _) ?! ]8 ]; I1 J/ C% p
大家好,我是侯小啾!0 m# n; ^& x( s8 J0 r" ^- `/ {
+ e! c3 k4 j. U! @, M* H6 l
 今天分享的话题是朴素贝叶斯分类器算法。3 G& E0 e% B5 _1 A" h: |$ I

- W, F) E" `& z. p🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ
3 r3 _* |+ d! z: y, C/ [$ V, B4 r2 l; o& m( J
1. 朴素贝叶斯算法原理6 k. X$ k- P" p
使用贝叶斯算法,首先需要理解的是以下两个公式:
0 ]: }3 s7 a- _' s
4 q( W8 P' s- X0 H1 G- x/ n' h# @全概率公式
+ Y1 w. v+ \1 E& H6 _' d: F. Y) ~* F# q7 e- @) f; ^2 i
     ( B ) = P ( A 1 ) × P ( B ∣ A 1 ) + P ( A 2 ) × P ( B ∣ A 2 ) + . . . + P ( A n ) × P ( B ∣ A n ) P(B)=P(A_1)×P(B|A_1)+P(A_2)×P(B|A_2)+...+P(A_n)×P(B|A_n)P(B)=P(A
1 {1 N4 R- r. Y; T6 E1! v: s1 y7 }" R2 v( H* m
6 }: S/ B8 m# [. G! P& l
)×P(B∣A + B. L6 R, s- o) d! G: s" T
1
1 ?' x+ g/ \, h' F
" d% l  \# V$ w: s6 {; v# I1 H )+P(A
! e. i" z7 U/ J8 F* _8 V2
/ J% J  C5 E, m& x  l& c: K! Y& F( e4 A) t5 n6 s
)×P(B∣A
9 ^; G0 V9 M  A& A, L( D6 f26 _/ Z2 x! t0 B: I; n% N

/ c7 e" @, j. t9 h, R% _  e )+...+P(A & }3 ^- u& c, i4 O
n
% i+ j3 D7 o. K: r6 u
9 P5 y* L  T' G& u1 j8 [ )×P(B∣A
% e! l5 j+ U4 bn
) b* t0 p  B- F8 s. t8 w1 ?( z5 f# R1 J9 U; ^8 q2 n
)" ^% b: b7 K- W* D
/ ^$ a, W# {$ f/ @5 Q4 c- h/ Y
贝叶斯公式
5 A" U; C' ~& x* N* @: e1 X* _" d( Y' a; p  x" K4 t6 E/ n
           ( A ∣ B ) = P(A|B)=P(A∣B)=P ( A ) × P ( B ∣ A ) P ( B ) \frac{P(A)×P(B|A)}{P(B)}
: m  T7 J4 w+ J8 U, pP(B): z; ^. G" R* B; }  u3 c
P(A)×P(B∣A)
3 e; y6 }! y* ^+ |0 o. ~! O# \0 j! N- |. V! W* Z

% }" C" v7 o* s# ]: y- d
6 |% p; o4 K  C            或
" e2 P) O# v- d, \4 _& g6 s5 B$ w( z& @4 I8 E8 M( L6 |
           ( A n ∣ B ) = P(A_n|B)=P(A
- |6 I' `4 r9 G9 y) z* ~7 k. s1 Dn/ |2 B$ F5 g! r3 Q; y2 o
( G  v. N+ f$ c' L: n, q% v
∣B)=P ( A n ) × P ( B ∣ A n ) P ( B ) \frac{P(A_n)×P(B|A_n)}{P(B)}
& F' O, r+ {$ j/ JP(B)1 Q" X/ I+ a9 ], N
P(A 8 A& T0 u1 f! s# O* w
n
$ x& j; o2 c0 i- u5 L4 q8 q& V/ R/ D; r% z% p$ h1 r8 f
)×P(B∣A
1 x& e$ w/ b- |+ _* h/ Sn
; B* T7 q. V: z3 Y  j' J
5 K: O) n, }/ Q- d0 Z- P, Q9 i' ^ )
& z  Q5 m; x+ O( B1 C8 {( A9 z: B! b$ [5 C

$ a4 D3 r+ s9 l* @2 {: Y- t# ?4 j& S
/ j" g/ z8 X/ L  T通俗地讲,假设需要将对某样本分为为0和1二类,其有A,B,C三个特征且值分别为a、b、c,
4 p3 h$ V# x, S, w则只需要求出,在类别为0前提下特证A、B、C分别为a、b、c的概率:
7 k8 ?; T: @$ ]" Z( R1 c+ I, H7 t7 }  m/ [, D$ i3 X
         即P ( A = a , B = b , C = c ∣ 类别为 0 ) P(A=a,B=b,C=c|类别为0)P(A=a,B=b,C=c∣类别为0)
# j8 v3 R/ x3 D+ S( a2 I  R
. g6 C: D+ Y. H1 g# X( H$ O' T6 K, q和在类别为1的前提下特征A、B、C分别为a、b、c的概率:' [/ w$ W$ X+ L" {1 Q
9 D6 F- b4 q  [* \: B( z$ {* X( M
         即P ( A = a , B = b , C = c ∣ 类别为 1 ) P(A=a,B=b,C=c|类别为1)P(A=a,B=b,C=c∣类别为1)8 x" X- M- \) N

3 \, a% W0 t9 `2 W/ ~- [. e即可。
3 E9 V* r+ s7 y. i+ V; C! ~' M然后将这两个概率作比较,大者,则为朴素贝叶斯决策结果所属类别。
$ N% E; \4 U* @& n' o4 Y  g5 h或者也可以求出在特征为目标特征的前提下,类别为某类别的概率,这样比较出的结果也是会是一致的。2 e9 R1 M. w- g/ j3 [

  k' k& B1 h/ r3 R) K( h. z% u2. sklearn提供的朴素贝叶斯算法( c; U, z7 d6 ?6 _) b
在python的scikit-learn库中,一共提供了3个朴素贝叶斯的分类算法,分别是GaussianNB(),MultinomialNB()和BernoulliNB()。
7 ], P# o0 f2 ^0 h8 w, m% ]( U2 x1 s" i0 @, {+ a3 u
其中,
! P. w" O* {- Q; E; B6 OBernoulliNB() 则表示先验为 伯努利分布 的朴素贝叶斯;; F- A5 ^4 y0 k4 h7 a: A, |
MultinomialNB() 表示先验为 多项式分布 的朴素贝叶斯;
' y2 ]+ X# g4 C/ [' `* o8 \: HGaussianNB() 表示先验为 高斯分布(即正态分布)的朴素贝叶斯。
/ o4 v, S8 `5 G' Y* O3 ?3 S7 F& t( y: C' C# k( N
伯努利朴素贝叶斯
4 H% }# C, n; w/ C( a其中伯努利分布的较为简单,因为伯努利分布的样本的特征值是离散型的分布,且特征都只有两个取值(比如0或1,是或否)。
' W& g  ]& }: R/ n( `多个特征,则对应多个相互独立的伯努利实验,每个实验只进行一次。比如,特征甲,可以是抛一枚硬币,结果可以是正面和反面。特征乙可以是昨天是否下雨,结果可以是“是”或“否”。
/ S! d1 z0 }7 L; B如果特征值是连续的,则算法中可以设定一个阈值(参数名为binarize),对特征值进行二值化处理,后即可满足伯努利分布。. R2 i9 \. R: c" \# N3 D2 q: i1 w3 J

2 x" E% M" |1 _6 }, f' j; c多项式朴素贝叶斯" p6 S4 \6 A; U" D$ Y" S! @, k
多项式分布则是在伯努利分布的基础上进行了多次实验。多项式分布下的多个特征依然分别对应着相互独立的实验,但与伯努利不同的是,每个特征的实验进行了多次。比如,特征甲可以是抛10次硬币正面的次数,特征乙可以是过去七天中下雨的天数(假设每天下雨的概率都一样)。
6 C/ ~* c# x4 l0 @2 I
6 |7 t: I; a4 W0 v7 z+ H高斯朴素贝叶斯6 S- x  M9 K* l, F- y
伯努利朴素贝叶斯算法和多项式朴素贝叶斯算法都是对于离散特征的,高斯朴素贝叶斯则常用于连续特征的情况下。高斯朴素贝叶斯假设特征满足高斯分布(正态分布)。如特征为“某地高中生的身高”时,就使用高斯朴素贝叶斯。
0 E2 n) l' d% C+ D
& ^6 o7 e0 X5 _! F3. 伯努利朴素贝叶斯 BernoulliNB()
: G" Z# q/ q4 @. \导包,并准备一组数据,代码如下图所示:: l( l! J- y8 g5 k+ f; w
/ R) Z; }9 Y" \0 k' j7 r) [
import numpy as np
( \0 Y$ S7 ~5 x" B1 f. C. sfrom sklearn.naive_bayes import BernoulliNB' _% N( t! `8 ?* ~2 t
from sklearn.datasets import make_blobs
" d& B8 y- m" L3 f4 Efrom sklearn.model_selection import train_test_split+ z* O/ I0 k- [# K1 N( w. \
. J* F% h, Y1 F: b4 Y( E+ p
# 500个样本,3个特征,3个标签,范围为(-10.0,10.0)。
( b( n' S& t/ c! Y; X+ ?/ FX, y = make_blobs(n_samples=500, n_features=3, centers=3, center_box=(-10.0, 10.0),random_state=10)
) s7 ]+ r$ B5 x2 f4 t/ q5 j; dprint(X)
* ?! @$ t3 _, y. {  sprint("=================================================================")
- x( f) H" a2 v7 bprint(y)/ V1 h. c# l, I' C/ _, G
1
4 ^: b6 p, y; ~! }8 C& y/ u: t7 r2# w8 b8 ^$ F/ d6 e) B
38 U3 K+ ?8 i' K8 v$ ?& ~
4
4 w5 x) I. z; w! I* a3 w. }5
$ m5 r6 u: S: _* P' q& K: K2 [$ I6
3 x7 G5 _4 E3 y: d6 @- h! @3 K7
% l# K& }1 I+ ^! e' ~$ Y  s8
% u: b4 e% q  g9% i- V% k  t$ w) d  Y
102 r7 c* o) q8 p# x; [- k5 q( n/ K+ ?
数据输出如下:
9 Y- Y2 C& l# N  e- }! W" d$ j( E+ p+ E: ~+ T
  s6 S* _/ K5 Y! n1 g& s% t1 W9 t
数据如图所示,可以看到,特征数据X的取值为连续的,因此要想使用伯努利朴素贝叶斯,需要先对特征数据做二值化处理。做二值化处理需要在实例化BernoulliNB()的时候,选择binarize属性。该属性默认值为0。
6 a  e; k0 u, Y3 f官方文档解释:8 \1 X, ]( g7 C9 G# E
# _/ E5 F& W( B: a' S
binarize : float or None, default=0.0- v0 b* m' p! Q* P
Threshold for binarizing (mapping to booleans) of sample features.$ ]6 x3 N. f5 ~8 k
If None, input is presumed to already consist of binary vectors.; S+ K+ w# n) ?& B: ^
4 u7 _' A! m0 K! d( t+ C1 Y+ J, n
binarize:浮点类型或None,默认值为0.0" ?  v, p. j: |. g3 g
样本特征二值化(映射到布尔值)的阈值。2 r- O) V2 i9 S" m! \+ G0 C) g* a
如果为None,则假定输入的特征数据已经是二值化的向量了。) u' l! N1 O+ D6 {  X3 b* R

5 o9 Y' R3 T9 D4 l( G训练模型,并使用测试集数据检验模型得分:
' W$ O0 k- Q: E  U" C7 i/ m  y5 g5 i- K" y& i* I
X_train,X_test,y_train,y_test=train_test_split(X, y, random_state=10)4 d7 |, u; O1 N4 q: O  N: x
nb = BernoulliNB()1 e4 G% G  P1 _: ?& e) i( \
nb.fit(X_train, y_train)7 w0 o, `/ p* e4 x7 }! c+ b  ?
print('模型得分:{:.3f}'.format(nb.score(X_test, y_test))). F( @" G% u! S3 ]) L
13 g! X0 O3 b; }' K# a
2
  c8 [5 {$ `* V9 k# O; _# B! f3
# e+ ?2 n4 {2 H* P4
, R5 t  W3 s0 P( R. J于是模型得到了一个“离谱”的得分1.0(简直不能更好):8 H/ z5 n5 t+ t5 d; w9 w
2 A2 s7 X) @0 u8 T& d2 i1 C3 w
6 i1 H/ @* }; }
输出预测结果:
' j; z5 D+ L! f$ Z# A
. V3 B2 r; ^2 y+ p; G: ~, d% s. Gpred = nb.predict(X_test)
/ ]1 }8 L. t: c7 N9 [) uprint(pred)
5 ?2 Q: f: w: A% J3 R# }) b& e. b* c5 h16 r9 d) o* \9 h. O+ h$ l
2
  m* h' W" P! k; v8 g/ x7 h2 `# N' Q+ a5 h- C1 d5 ]
0 r  H3 R/ ]4 f; C* S1 w* P3 p! V
4. 多项式朴素贝叶斯 MultinomialNB()
7 C# r) U' D4 ?, q! J  t8 iMultinomialNB()实现了服从多项分布数据的朴素贝叶斯算法。( \( v; Y0 U7 \+ d6 y2 P
也是用于文本分类的两大经典朴素贝叶斯算法之一(文本分类领域中数据往往以词向量表示,尽管在实践中 tf-idf 向量在预测时表现良好)。
! G! [7 z9 O9 L  w9 ?4 R$ u
. Y+ K1 D  z4 V4 X生成两组随机数,用于示例展示。一组特征值,一组标签。' q8 K4 Y* L3 r% `9 U
# n. \$ w& l' i) o1 l
import numpy as np/ T2 {6 i% \: x& J/ O/ c7 J) D* ]
from sklearn.naive_bayes import MultinomialNB- Q: v2 ~' C) E  h9 P$ V. x' Q9 _
from sklearn.model_selection import train_test_split
6 l% n) c4 y: T: M/ ~
1 N3 f4 r/ }3 l- d. y7 l6 K  \4 d1 J+ C) ^4 c
# 设置随机数种子
5 C) @* M. h: S( Qnp.random.seed(10)5 z3 H9 t  X! ?! d, x
# 1000个样本,5个特征,每个样本的取值是[0,9]的整数
: }( W, Y' p& I4 ^  YX = np.random.randint(10, size=(1000, 5))7 J$ @/ |- u* A/ Q2 e
# 标签可以有的取值为0,1,28 d. t2 |. K6 \' g! H
y = np.random.randint(3, size=[1000])# R( L3 J) m+ k
print(X)  |4 U  \: f& j% d
print(y): |3 l; L3 Q1 a0 p* ^0 e# e' h3 d
19 m) \# z* _+ X' ]7 @; Z2 L* a  N
2. X5 G9 T9 c: G( Y
3( T( l  t# J; h& M0 d. d
4. D; [) s1 h( l" n
5) t+ `+ r3 q; A
6- G; C1 R% w+ T& G# d$ y$ ^/ Q
7
. B' D5 Q- V+ \  b9 Q- h8
+ F( A- s% h, j% b8 s8 K+ ^. F9
+ x6 O. X6 _! l- t107 X) f0 M  g( o1 n
11
0 h6 E2 t8 S# v, C% c% M12
* C' I6 y+ `) m0 u5 k135 v+ s( w$ y- M. b6 i: ^9 O8 ?
部分数据展示如下:
- W4 j8 A' d0 |, q& r, o- W; N( q' M; r9 V! {6 M
9 f+ U+ p8 g# L; H
# 分割数据
# _8 p( o9 W( u) `X_train, X_test, y_train, y_test = train_test_split(X, y, random_state=10)& B  L( [' e$ w7 ?6 i
# 训练多项式朴素贝叶斯模型* L6 A( v, _$ |9 Q+ G/ ~0 R
nb = MultinomialNB()
4 c. e3 O( d" z7 k2 @nb.fit(X_train, y_train)$ J- \5 u8 p9 q1 ]7 N
4 M. L' o$ O0 f" U, e
print('模型得分:{:.3f}'.format(nb.score(X_test, y_test)))
+ R% X8 W' y" `5 K6 p) i' Ppred = nb.predict(X_test)0 n0 E7 \+ Q1 x4 y" w0 `$ c
print(pred); w) ~* {0 o) t) W# F- S
1+ b. V- G; W0 u. o- A
2; h( S5 y) ?7 e/ m3 G
3
& O, Q/ i0 B' ?4
& w3 g9 ]  e! ]9 J5$ [2 Q* S( ~6 [/ S! p% c5 {8 C
69 f3 D, W: M4 h6 \& h% c$ T
7% v5 c% D. k( ^, W1 [4 E
8. m$ ]/ u* |1 Q9 Q/ d5 R
92 H% y: Z6 g( S, b
模型得分及预测结果如下图所示:
1 l) B) L) ]2 T* J9 d) Z! D, R: A8 D
因为数据是随机生成的,所以这里的特征与标签之间无逻辑关系可言,以至于这里的模型得分偏低。但是算法是有效的。
" D3 D- ^& A& y/ E5 n- _* ?; ?  ?+ }. O7 k# p* m, _9 J
5. 高斯朴素贝叶斯 GaussianNB()( S4 u/ V- }  Y8 ^7 W3 U
符合高斯分布的数据也是连续的数值,所以不像伯努利分布的特征数据那样直接可以直接计算概率,但是也不是要做二值化处理。因为其符合正态分布,所以可以直接从正态分布中找到其概率值,
6 D) J7 q% p# ?- p3 z8 d  n
5 Z8 K6 H' Q+ \3 G5 L. ~, T+ E假设以某学校同学的身高数据为例,其中男生身高满足均值为176,标准差差为10的正态分布,则已知某同学为男生,其身高为180的概率为:) _, N2 T. i. d2 v1 T
+ A3 K/ |/ i2 J3 q( H6 Q
print(stats.norm.pdf(180, 174, 6)), @9 }4 L4 y( i8 w! {
1
4 ?$ ^! ^* B! T/ b; {
8 e: E0 m# y4 C, h这个概率值虽然不大,但是如果与女生的数据相比,若女生的身高为180的概率小于0.0368,则由此可以判断出该同学为男生。
) i" c  h$ m+ o0 [4 B2 g. N; U+ f) V1 L, P3 A2 ~, J' U7 W
根据这样原理,我们使用sklearn库做如下高斯贝叶斯代码示例:
) o. d# T. W& U( L' b  o7 w2 h( b' c: b) w7 ^
import numpy as np6 k' S' _$ M8 Y- y1 D9 U7 v- X
from sklearn.naive_bayes import GaussianNB# \* r) N9 P: C
from sklearn.model_selection import train_test_split. L) m7 s. q+ n. i1 ^
4 l( m) C6 }" g7 i+ E
0 c' R- Y  A: u% H
# 设置随机数种子
+ l8 m7 L# V/ U/ Jnp.random.seed(10)2 h. ^2 C& ^/ P, h2 Z3 W
# 特征值:均值为4,标准差为5的正态分布,10000个样本,3个特征9 d. U( \' S( b* w4 H+ z
X = np.random.normal(4, 5, (10000, 3))' S* {/ d1 ~# v5 @/ x
# 标签值:0,1,2
8 k  n) I7 ^# l6 py = np.random.randint(3, size=[10000])
  H8 k" P2 K0 s2 Q
  L  O# I1 H/ K, ]! r- Z2 Tprint(X)
) Z- k) |  T" W! b# dprint(y)( m) f1 F9 H  {- R8 a9 L
1
7 k; z5 A0 U. O. m8 |2  v( b! k; t) M- v5 X2 F
3+ {5 F; X3 N! D$ F
4
9 a$ u5 ]# x( j& a6 x5
$ `0 H: v2 o: `! B) E6* h6 g( B7 Y1 [/ ?# ?! V
7: ~# m& T6 j* G. u* S  `' O' q5 S8 D
8/ l5 w, x; }1 K4 K4 I& y/ |
9: F5 r6 X: N1 m( r* r$ S; K7 N& p
10
1 u2 p0 s# G4 T" s& M: R: a11+ z( u% I& f; X& R8 R+ f6 g. p  F
12" Y$ l0 R  H: J$ n  L
132 @$ H$ o& l: y9 B3 P
14
* I9 P: c. ?0 I- E: {/ J2 B生成数据如下图所示:
$ B0 Q. v! s9 A+ T3 k7 o$ Z+ U+ j6 ~0 U
+ `7 n& X% @, W: n
X_train, X_test, y_train, y_test = train_test_split(X, y, random_state=10)  z* P: m1 z3 v% j; }3 v
nb = GaussianNB()
, Y  d4 r2 n( j; Fnb.fit(X_train, y_train)( @7 H2 z9 l2 i# B* x

- D" f5 X3 J, E  n$ b0 M( Oprint('模型得分:{:.3f}'.format(nb.score(X_test, y_test)))
" L; e! q7 T: s  J; e5 g' j: d6 I$ z' q" P& l8 r
pred = nb.predict(X_test)
" v# Q% a, ~9 _8 ]! C7 Kprint(pred)4 o7 k+ c, i$ t: d2 V' p  p* H
————————————————- e( e: M% p1 u( A8 Z' W
版权声明:本文为CSDN博主「侯小啾」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。$ |2 D+ ]! c7 K0 _' l. L: h
原文链接:https://blog.csdn.net/weixin_48964486/article/details/126337511# l7 Z( W/ _' Y

; H9 ^. B8 Z( Z$ h; k5 r* q8 A4 {# \% `  ~* J% ~" w! M$ D





欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5