找回密码
 会员注册
查看: 46|回复: 0

ML朴素贝叶斯分类器及Python手写实现(详细)

[复制链接]

2万

主题

0

回帖

7万

积分

超级版主

积分
73515
发表于 2024-9-3 23:33:53 | 显示全部楼层 |阅读模式
目录一、两个学派1.1频率学派:1.2贝叶斯学派二、朴素贝叶斯2.1简介2.2相关公式2.2.1贝叶斯公式2.2.2朴素贝叶斯公式三、案例分析(西瓜集)四、Python实现NB代码解读4.1西瓜集手写代码4.2鸢尾花手写代码4.3鸢尾花第三方库 五、Python实现NB代码汇总5.1不使用sklearn中方法5.1.1数据集:西瓜集3.0(来自西瓜书《机器学习周志华》84页)5.1.2数据集:鸢尾花iris(来自sklearn提供的数据集)5.2使用sklearn中的方法5.2.1数据集:鸢尾花iris(来自sklearn提供的数据集)【写在前面】本篇文章介绍了朴素贝叶斯分类器(NaiveBayes,NB)的工作原理以及使用Python实现NB,这里介绍了3种代码:使用西瓜集3.0手写NB、使用鸢尾花数据集手写NB、使用鸢尾花数据集使用机器学习第三方库实现NB。如有错误欢迎指正。一、两个学派概率模型的训练过程就是参数估计过程。对于参数估计,统计学界的两个学派(频率学派、贝叶斯学派)提供了不同的解决方案。1.1频率学派:核心思想:认为参数真值是固定的、未知的一个常数,而观察到的数据是随机的。其着眼点是样本空间。解决方法:最大似然估计(MLE)。1.2贝叶斯学派核心思想:认为参数真值是未观察到的随机变量,其本身也可有分布,而观察到的数据是固定的。其着眼点是参数空间,重视参数的分布。利用参数的先验信息估计后验概率,即先验概率P(c)+样本信息(条件概率)P(x|c)->后验概率P(c|x)。解决方法:最大后验估计(MAP)。频率学派的观点是对总体分布做适当的假定,结合样本信息对参数进行统计推断,这里涉及总体信息和样本信息;而贝叶斯学派的观点认为除了上述两类信息之外,统计推断还应引入先验信息。一般来说,先验信息来源于经验和历史资料。频率学派不假设任何的先验信息,不参照过去的经验,只按照当前已有的数据进行概率推断;贝叶斯学派会假设先验信息的存在。二、朴素贝叶斯2.1简介贝叶斯分类的分类原理就是利用贝叶斯公式根据某特征的先验概率计算出后验概率,然后选择具有最大后验概率的类别作为该特征所属的类别。朴素贝叶斯之所以称为“朴素”,是因为假设了各个特征之间是相互独立的。2.2相关公式对于待分类样本S:        特征集合X={x1,x2,…,xn}        类标签集合C={c1,c2,…,cm}表示样本S具有n个特征,可能会有m中不同的分类结果。2.2.1贝叶斯公式其中:ci:表示一种预测类别结果;X:表示样本S的一组特征观测数据;P(ci):表示ci的先验概率,即在未观察到X之前S属于ci的概率;P(ci|X):表示ci的后验概率;即在观察到X后S属于ci的概率;P(X|ci)/P(X):可能性函数,表示调整因子,使预测概率更加接近真实概率;所以,贝叶斯公式:后验概率=先验概率*调整因子。调整因子>1,则表示先验概率被增强,事件ci发生的可能性变大。调整因子=1,则表示数据X对判断事件ci发生的概率没有帮助。调整因子6.80*10­^(-5),所以朴素贝叶斯分类器将待测样本判别为“好瓜”。 四、Python实现NB代码解读4.1西瓜集手写代码(1)加载第三方模块importnumpyasnpimportmathimportpandasaspd(2)定义加载数据集函数defloadDataSet():dataSet=[['青绿','蜷缩','浊响','清晰','凹陷','硬滑',0.697,0.460,'好瓜'],['乌黑','蜷缩','沉闷','清晰','凹陷','硬滑',0.774,0.376,'好瓜'],['乌黑','蜷缩','浊响','清晰','凹陷','硬滑',0.634,0.264,'好瓜'],['青绿','蜷缩','沉闷','清晰','凹陷','硬滑',0.608,0.318,'好瓜'],['浅白','蜷缩','浊响','清晰','凹陷','硬滑',0.556,0.215,'好瓜'],['青绿','稍蜷','浊响','清晰','稍凹','软粘',0.403,0.237,'好瓜'],['乌黑','稍蜷','浊响','稍糊','稍凹','软粘',0.481,0.149,'好瓜'],['乌黑','稍蜷','浊响','清晰','稍凹','硬滑',0.437,0.211,'好瓜'],['乌黑','稍蜷','沉闷','稍糊','稍凹','硬滑',0.666,0.091,'坏瓜'],['青绿','硬挺','清脆','清晰','平坦','软粘',0.243,0.267,'坏瓜'],['浅白','硬挺','清脆','模糊','平坦','硬滑',0.245,0.057,'坏瓜'],['浅白','蜷缩','浊响','模糊','平坦','软粘',0.343,0.099,'坏瓜'],['青绿','稍蜷','浊响','稍糊','凹陷','硬滑',0.639,0.161,'坏瓜'],['浅白','稍蜷','沉闷','稍糊','凹陷','硬滑',0.657,0.198,'坏瓜'],['乌黑','稍蜷','浊响','清晰','稍凹','软粘',0.360,0.370,'坏瓜'],['浅白','蜷缩','浊响','模糊','平坦','硬滑',0.593,0.042,'坏瓜'],['青绿','蜷缩','沉闷','稍糊','稍凹','硬滑',0.719,0.103,'坏瓜']]testSet=['青绿','蜷缩','浊响','清晰','凹陷','硬滑',0.697,0.460]#待测集labels=['色泽','根蒂','敲声','纹理','脐部','触感','密度','含糖率']#特征returndataSet,testSet,labelsdataSet:训练集;testSet:待测集;labels:样本所具有的特征的名称; (3)计算连续属性的均值和标准差#?计算(不同类别中指定连续特征的)均值、标准差defmean_std(feature,cla):dataSet,testSet,labels=loadDataSet()lst=[item[labels.index(feature)]foritemindataSetifitem[-1]==cla]#类别为cla中指定特征feature组成的列表mean=round(np.mean(lst),3)#均值std=round(np.std(lst),3)#标准差returnmean,stdfeature:传入指定将要计算其均值的标准差的特征名称;cla:传入条件概率的条件,计算指定分类cla下该特征的条件概率;函数返回计算结果:均值mean、标准差std;(4)计算先验概率#?计算先验概率P(c)defprior():dataSet=loadDataSet()[0]#载入数据集countG=0#初始化好瓜数量countB=0#初始化坏瓜数量countAll=len(dataSet)foritemindataSet:#统计好瓜个数ifitem[-1]=="好瓜":countG+=1foritemindataSet:#统计坏瓜个数ifitem[-1]=="坏瓜":countB+=1#计算先验概率P(c)P_G=round(countG/countAll,3)P_B=round(countB/countAll,3)returnP_G,P_BcountG:统计训练集中好瓜的数量;countB:统计训练集中坏瓜的数量;countAll:训练集中所有样本的数量;P_G:好瓜的先验概率;P_B:坏瓜的先验概率;函数返回计算出的先验概率P_G、P_B; (5)计算离散属性的条件概率#?计算离散属性的条件概率P(xi|c)defP(index,cla):dataSet,testSet,labels=loadDataSet()#载入数据集countG=0#初始化好瓜数量countB=0#初始化坏瓜数量foritemindataSet:#统计好瓜个数ifitem[-1]=="好瓜":countG+=1foritemindataSet:#统计坏瓜个数ifitem[-1]=="坏瓜":countB+=1lst=[itemforitemindataSetif(item[-1]==cla)&(item[index]==testSet[index])]#lst为cla类中第index个属性上取值为xi的样本组成的集合P=round(len(lst)/(countGifcla=="好瓜"elsecountB),3)#计算条件概率returnPindex:用于根据待测集样本中特征值的索引位置在训练集中样本的对应索引位置上进行遍历;cla:用于在训练集中限定样本的类别,便于计算离散属性各特征的条件概率;countG:统计训练集中好瓜的数量;countB:统计训练集中坏瓜的数量;lst:为cla类别中第index个属性上取值为xi的样本组成的集合;P:离散属性的条件概率;函数返回计算出的条件概率P;  (6)计算连续属性的条件概率#?计算连续属性的条件概率p(xi|c)defp():dataSet,testSet,labels=loadDataSet()#载入数据集denG_mean,denG_std=mean_std("密度","好瓜")#好瓜密度的均值、标准差denB_mean,denB_std=mean_std("密度","坏瓜")#坏瓜密度的均值、标准差sugG_mean,sugG_std=mean_std("含糖率","好瓜")#好瓜含糖率的均值、标准差sugB_mean,sugB_std=mean_std("含糖率","坏瓜")#坏瓜含糖率的均值、标准差#p(密度|好瓜)p_density_G=(1/(math.sqrt(2*math.pi)*denG_std))*np.exp(-(((testSet[labels.index("密度")]-denG_mean)**2)/(2*(denG_std**2))))p_density_G=round(p_density_G,3)#p(密度|坏瓜)p_density_B=(1/(math.sqrt(2*math.pi)*denB_std))*np.exp(-(((testSet[labels.index("密度")]-denB_mean)**2)/(2*(denB_std**2))))p_density_B=round(p_density_B,3)#p(含糖率|好瓜)p_sugar_G=(1/(math.sqrt(2*math.pi)*sugG_std))*np.exp(-(((testSet[labels.index("含糖率")]-sugG_mean)**2)/(2*(sugG_std**2))))p_sugar_G=round(p_sugar_G,3)#p(含糖率|坏瓜)p_sugar_B=(1/(math.sqrt(2*math.pi)*sugB_std))*np.exp(-(((testSet[labels.index("含糖率")]-sugB_mean)**2)/(2*(sugB_std**2))))p_sugar_B=round(p_sugar_B,3)returnp_density_G,p_density_B,p_sugar_G,p_sugar_BdenG_mean:好瓜密度的均值;denG_std:好瓜密度的标准差;denB_mean:坏瓜密度的均值;denB_std:坏瓜密度的标准差;sugG_mean:好瓜含糖率的均值;sugG_std:好瓜含糖率的标准差;sugB_mean:坏瓜含糖率的均值;sugB_std:坏瓜含糖率的标准差;p_density_G:好瓜条件下对于密度这一特征的条件概率;p_density_B:坏瓜条件下对于密度这一特征的条件概率;p_sugar_G:好瓜条件下对于含糖率这一特征的条件概率;p_sugar_B:坏瓜条件下对于含糖率这一特征的条件概率;函数返回连续属性的条件概率; (7)计算后验概率#?预测后验概率P(c|xi)defbayes():#?计算类先验概率P_G,P_B=prior()#?计算离散属性的条件概率P0_G=P(0,"好瓜")#P(青绿|好瓜)P0_B=P(0,"坏瓜")#P(青绿|坏瓜)P1_G=P(1,"好瓜")#P(蜷缩|好瓜)P1_B=P(1,"坏瓜")#P(蜷缩|好瓜)P2_G=P(2,"好瓜")#P(浊响|好瓜)P2_B=P(2,"坏瓜")#P(浊响|好瓜)P3_G=P(3,"好瓜")#P(清晰|好瓜)P3_B=P(3,"坏瓜")#P(清晰|好瓜)P4_G=P(4,"好瓜")#P(凹陷|好瓜)P4_B=P(4,"坏瓜")#P(凹陷|好瓜)P5_G=P(5,"好瓜")#P(硬滑|好瓜)P5_B=P(5,"坏瓜")#P(硬滑|好瓜)#?计算连续属性的条件概率p_density_G,p_density_B,p_sugar_G,p_sugar_B=p()#?计算后验概率isGood=P_G*P0_G*P1_G*P2_G*P3_G*P4_G*P5_G*p_density_G*p_sugar_G#计算是好瓜的后验概率isBad=P_B*P0_B*P1_B*P2_B*P3_B*P4_B*P5_B*p_density_B*p_sugar_B#计算是坏瓜的后验概率returnisGood,isBad 调用prior()函数计算先验概率;调用P()函数计算离散属性的条件概率;调用p()函数计算连续属性的条件概率;isGood:好瓜的后验概率;isBad:坏瓜的后验概率;函数返回后验概率的计算结果isGood、isBad;(8)主函数调用上述函数模块if__name__=='__main__':dataSet,testSet,labels=loadDataSet()testSet=[testSet]df=pd.DataFrame(testSet,columns=labels,index=[1])print("=======================待测样本========================")print(f"待测集:\n{df}")isGood,isBad=bayes()print("=======================后验概率========================")print("后验概率:")print(f"P(好瓜|xi)={isGood}")print(f"P(好瓜|xi)={isBad}")print("=======================预测结果========================")print("predict--->好瓜"if(isGood>isBad)else"predict--->坏瓜")运行结果:从上图的终端输出结果可以看到待测样本的各个属性值,然后可以看到后验概率的计算结果,最后是预测结果,此样本的预测结果为“好瓜”。此训练集与待测样本是“朴素贝叶斯西瓜集分类案例”的数据,通过比较最终的两个预测结果一致,也可表明朴素贝叶斯分类器编写代码基本正确。4.2鸢尾花手写代码参考4.1运行结果:4.3鸢尾花第三方库(1)加载第三方模块importpandasaspdfromsklearn.datasetsimportload_irisfromsklearn.naive_bayesimportGaussianNBfromsklearn.model_selectionimporttrain_test_splitfromsklearn.metricsimportaccuracy_score因为鸢尾花数据集中各特征为连续特征值,所以需要导入sklearn库中的高斯朴素贝叶斯分类器GaussianNB; 下面通过定义了一个bayes_model类#?定义朴素贝叶斯分类器classbayes_model():def__int__(self):pass(2)定义加载数据函数#?加载数据defload_data(self):data=load_iris()iris_data=data.datairis_target=data.targetiris_featureNames=data.feature_namesiris_features=pd.DataFrame(data=iris_data,columns=iris_featureNames)train_x,test_x,train_y,test_y=train_test_split(iris_features,iris_target,test_size=0.3,random_state=123)returntrain_x,test_x,train_y,test_ydata:加载sklearn库中的鸢尾花数据集;iris_data:加载鸢尾花数据集的属性值;iris_target:加载鸢尾花数据集的分类标签;iris_featureNames:加载数据的特征名称;train_test_split()函数:将数据集划分为训练集和测试集;函数返回划分好的训练集和测试集;(3)训练高斯朴素贝叶斯分类器#?训练高斯朴素贝叶斯模型deftrain_model(self,train_x,train_y):clf=GaussianNB()clf.fit(train_x,train_y)returnclfclf:定义一个高斯朴素贝叶斯分类器(因为鸢尾花数据集中特征值为连续值);clf.fit():表示对高斯朴素贝叶斯分类器进行训练;函数返回一个训练好的高斯朴素贝叶斯分类器;(4)处理预测数据,进行终端输出#?处理预测的数据defproba_data(self,clf,test_x,test_y):y_predict=clf.predict(test_x)#返回待预测样本的预测结果(所属类别)y_proba=clf.predict_proba(test_x)#返回预测样本属于各标签的概率accuracy=accuracy_score(test_y,y_predict)*100#计算predict预测的准确率returny_predict,y_proba,accuracyy_predict:对测试样本进行类别预测,预测结果是一个确定的分类类别;y_proba:预测测试样本属于各类别的概率,预测结果是一个概率序列;accuracy:计算训练好的高斯朴素贝叶斯分类器对测试集的分类准确率;(5)训练数据#?训练数据defexc_p(self):train_x,test_x,train_y,test_y=self.load_data()#加载数据clf=self.train_model(train_x,train_y)#训练高斯朴素贝叶斯模型clfy_predict,y_proba,accuracy=self.proba_data(clf,test_x,test_y)#利用训练好的模型clf对测试集test_x进行结果预测分析returntrain_x,test_x,train_y,test_y,y_predict,y_proba,accuracy调用load_data()函数:加载数据,训练集和测试集;调用train_model()函数:根据巡礼那及训练一个高斯朴素贝叶斯分类器;调用proba_data()函数:利用训练好的分类器对测试集进行分类;(6)主函数调用上述模块if__name__=='__main__':train_x,test_x,train_y,test_y,y_predict,y_proba,accuracy=bayes_model().exc_p()#训练集与其标签df1df1_1=pd.DataFrame(train_x).reset_index(drop=True)df1_2=pd.DataFrame(train_y)df1=pd.merge(df1_1,df1_2,left_index=True,right_index=True)df1.columns=['sepallength(cm)','sepalwidth(cm)','petallength(cm)','petalwidth(cm)','trainclassify']print("=============================================训练集==============================================")print(f'ThetraindataSetis:\n{df1}\n')#测试集与其标签df2df2_1=pd.DataFrame(test_x).reset_index(drop=True)df2_2=pd.DataFrame(test_y)df2=pd.merge(df2_1,df2_2,left_index=True,right_index=True)df2.columns=['sepallength(cm)','sepalwidth(cm)','petallength(cm)','petalwidth(cm)','testclassify']print("=============================================测试集==============================================")print(f'ThetestdataSetis:\n{df2}\n')#预测结果tot1=pd.DataFrame([test_y,y_predict]).Ttot2=pd.DataFrame(y_proba).applymap(lambdax:'%.2f'%x)tot=pd.merge(tot1,tot2,left_index=True,right_index=True)tot.columns=['y_true','y_predict','predict_0','predict_1','predict_2']print("============================================预测结果==============================================")print('Theresultofpredictis:\n',tot)print("=============================================准确率==============================================")print(f'TheaccuracyofTestsetis:{accuracy:.2f}%')df1:训练集的二维结构;df2:测试集的二维结构;tot:预测结果的二维结构;运行结果:终端输出的训练集:训练集中有105的样本,第1~5列数值分别表示为花萼长度、花萼宽度、花瓣长度、花瓣宽度、所属类别。终端输出的测试集:测试集中共有45个样本。终端输出的准确率和预测结果:从上图可以看到最终训练好的高斯朴素贝叶斯分类器对测试集中45个样本的分类准确率高达95%,也就是说45个样本中只有2个样本分类错误,其余全部分类正确。预测结果是一个二维矩阵的呈现形式,每一行代表一个样本的预测信息,第1列表示测试集中样本的正确标签,第2列为预测的标签,第3~5列表示预测结果为三个类别的可能概率(概率和为1); 五、Python实现NB代码汇总5.1不使用sklearn中方法5.1.1数据集:西瓜集3.0(来自西瓜书《机器学习周志华》84页)#!朴素贝叶斯分类器_西瓜集分类importnumpyasnpimportmathimportpandasaspd#?加载数据集defloadDataSet():dataSet=[['青绿','蜷缩','浊响','清晰','凹陷','硬滑',0.697,0.460,'好瓜'],['乌黑','蜷缩','沉闷','清晰','凹陷','硬滑',0.774,0.376,'好瓜'],['乌黑','蜷缩','浊响','清晰','凹陷','硬滑',0.634,0.264,'好瓜'],['青绿','蜷缩','沉闷','清晰','凹陷','硬滑',0.608,0.318,'好瓜'],['浅白','蜷缩','浊响','清晰','凹陷','硬滑',0.556,0.215,'好瓜'],['青绿','稍蜷','浊响','清晰','稍凹','软粘',0.403,0.237,'好瓜'],['乌黑','稍蜷','浊响','稍糊','稍凹','软粘',0.481,0.149,'好瓜'],['乌黑','稍蜷','浊响','清晰','稍凹','硬滑',0.437,0.211,'好瓜'],['乌黑','稍蜷','沉闷','稍糊','稍凹','硬滑',0.666,0.091,'坏瓜'],['青绿','硬挺','清脆','清晰','平坦','软粘',0.243,0.267,'坏瓜'],['浅白','硬挺','清脆','模糊','平坦','硬滑',0.245,0.057,'坏瓜'],['浅白','蜷缩','浊响','模糊','平坦','软粘',0.343,0.099,'坏瓜'],['青绿','稍蜷','浊响','稍糊','凹陷','硬滑',0.639,0.161,'坏瓜'],['浅白','稍蜷','沉闷','稍糊','凹陷','硬滑',0.657,0.198,'坏瓜'],['乌黑','稍蜷','浊响','清晰','稍凹','软粘',0.360,0.370,'坏瓜'],['浅白','蜷缩','浊响','模糊','平坦','硬滑',0.593,0.042,'坏瓜'],['青绿','蜷缩','沉闷','稍糊','稍凹','硬滑',0.719,0.103,'坏瓜']]testSet=['青绿','蜷缩','浊响','清晰','凹陷','硬滑',0.697,0.460]#待测集labels=['色泽','根蒂','敲声','纹理','脐部','触感','密度','含糖率']#特征returndataSet,testSet,labels#?计算(不同类别中指定连续特征的)均值、标准差defmean_std(feature,cla):dataSet,testSet,labels=loadDataSet()lst=[item[labels.index(feature)]foritemindataSetifitem[-1]==cla]#类别为cla中指定特征feature组成的列表mean=round(np.mean(lst),3)#均值std=round(np.std(lst),3)#标准差returnmean,std#?计算先验概率P(c)defprior():dataSet=loadDataSet()[0]#载入数据集countG=0#初始化好瓜数量countB=0#初始化坏瓜数量countAll=len(dataSet)foritemindataSet:#统计好瓜个数ifitem[-1]=="好瓜":countG+=1foritemindataSet:#统计坏瓜个数ifitem[-1]=="坏瓜":countB+=1#计算先验概率P(c)P_G=round(countG/countAll,3)P_B=round(countB/countAll,3)returnP_G,P_B#?计算离散属性的条件概率P(xi|c)defP(index,cla):dataSet,testSet,labels=loadDataSet()#载入数据集countG=0#初始化好瓜数量countB=0#初始化坏瓜数量foritemindataSet:#统计好瓜个数ifitem[-1]=="好瓜":countG+=1foritemindataSet:#统计坏瓜个数ifitem[-1]=="坏瓜":countB+=1lst=[itemforitemindataSetif(item[-1]==cla)&(item[index]==testSet[index])]#lst为cla类中第index个属性上取值为xi的样本组成的集合P=round(len(lst)/(countGifcla=="好瓜"elsecountB),3)#计算条件概率returnP#?计算连续属性的条件概率p(xi|c)defp():dataSet,testSet,labels=loadDataSet()#载入数据集denG_mean,denG_std=mean_std("密度","好瓜")#好瓜密度的均值、标准差denB_mean,denB_std=mean_std("密度","坏瓜")#坏瓜密度的均值、标准差sugG_mean,sugG_std=mean_std("含糖率","好瓜")#好瓜含糖率的均值、标准差sugB_mean,sugB_std=mean_std("含糖率","坏瓜")#坏瓜含糖率的均值、标准差#p(密度|好瓜)p_density_G=(1/(math.sqrt(2*math.pi)*denG_std))*np.exp(-(((testSet[labels.index("密度")]-denG_mean)**2)/(2*(denG_std**2))))p_density_G=round(p_density_G,3)#p(密度|坏瓜)p_density_B=(1/(math.sqrt(2*math.pi)*denB_std))*np.exp(-(((testSet[labels.index("密度")]-denB_mean)**2)/(2*(denB_std**2))))p_density_B=round(p_density_B,3)#p(含糖率|好瓜)p_sugar_G=(1/(math.sqrt(2*math.pi)*sugG_std))*np.exp(-(((testSet[labels.index("含糖率")]-sugG_mean)**2)/(2*(sugG_std**2))))p_sugar_G=round(p_sugar_G,3)#p(含糖率|坏瓜)p_sugar_B=(1/(math.sqrt(2*math.pi)*sugB_std))*np.exp(-(((testSet[labels.index("含糖率")]-sugB_mean)**2)/(2*(sugB_std**2))))p_sugar_B=round(p_sugar_B,3)returnp_density_G,p_density_B,p_sugar_G,p_sugar_B#?预测后验概率P(c|xi)defbayes():#?计算类先验概率P_G,P_B=prior()#?计算离散属性的条件概率P0_G=P(0,"好瓜")#P(青绿|好瓜)P0_B=P(0,"坏瓜")#P(青绿|坏瓜)P1_G=P(1,"好瓜")#P(蜷缩|好瓜)P1_B=P(1,"坏瓜")#P(蜷缩|好瓜)P2_G=P(2,"好瓜")#P(浊响|好瓜)P2_B=P(2,"坏瓜")#P(浊响|好瓜)P3_G=P(3,"好瓜")#P(清晰|好瓜)P3_B=P(3,"坏瓜")#P(清晰|好瓜)P4_G=P(4,"好瓜")#P(凹陷|好瓜)P4_B=P(4,"坏瓜")#P(凹陷|好瓜)P5_G=P(5,"好瓜")#P(硬滑|好瓜)P5_B=P(5,"坏瓜")#P(硬滑|好瓜)#?计算连续属性的条件概率p_density_G,p_density_B,p_sugar_G,p_sugar_B=p()#?计算后验概率isGood=P_G*P0_G*P1_G*P2_G*P3_G*P4_G*P5_G*p_density_G*p_sugar_G#计算是好瓜的后验概率isBad=P_B*P0_B*P1_B*P2_B*P3_B*P4_B*P5_B*p_density_B*p_sugar_B#计算是坏瓜的后验概率'''print(P_G)print(P_B)print(P0_G)print(P0_B)print(P1_G)print(P1_B)print(P2_G)print(P2_B)print(P3_G)print(P3_B)print(P4_G)print(P4_B)print(P5_G)print(P5_B)print(p_density_G)print(p_density_B)print(p_sugar_G)print(p_sugar_B)'''returnisGood,isBadif__name__=='__main__':dataSet,testSet,labels=loadDataSet()testSet=[testSet]df=pd.DataFrame(testSet,columns=labels,index=[1])print("=======================待测样本========================")print(f"待测集:\n{df}")isGood,isBad=bayes()print("=======================后验概率========================")print("后验概率:")print(f"P(好瓜|xi)={isGood}")print(f"P(好瓜|xi)={isBad}")print("=======================预测结果========================")print("predict--->好瓜"if(isGood>isBad)else"predict--->坏瓜")5.1.2数据集:鸢尾花iris(来自sklearn提供的数据集)#!朴素贝叶斯分类器_鸢尾花分类importnumpyasnpfromsklearn.datasetsimportload_irisfromsklearn.model_selectionimporttrain_test_splitfromsklearn.metricsimportaccuracy_scoreiris=load_iris()X=iris.datay=iris.targetX_train,X_test,y_train,y_test=train_test_split(X,y,test_size=0.3,random_state=1,stratify=y)#?计算先验概率p(c)#存储在p_c列表内p_c=[p(0),p(1),p(2)]defprior():p_c=[]forcinnp.unique(y_train):tmp=[xforxiny_trainifx==c]p_c.append(len(tmp)/len(y_train))returnp_c#?计算条件概率p(xi|c)#存储在p_xc列表内p_xc=[p(x1|c),p(x2|c),p(x3|c),p(x4|c)]defp_xc(x,cla):tar=[X_train[i]fori,cinenumerate(y_train)ifc==cla]mean=np.mean(tar,axis=0)#均值var=np.var(tar,axis=0)#方差p_xc=np.exp(-(x-mean)**2/(2*var))/np.sqrt(2*np.pi*var)returnp_xc#?根据后验概率p(ci|x)预测结果#后验概率存储在proba列表内proba=[p(0|x),p(1|x),p(2|x)]defpredict():pred=[]#存放预测结果forxinX_test:proba=[]#存放属于各个类的概率(后验概率)fori,cinenumerate(np.unique(y_train))_c=prior()P_xc=p_xc(x,c)proba.append(P_c[i]*np.prod(P_xc))res=np.unique(y_train)[proba.index(max(proba))]pred.append(res)returnnp.array(pred)y_pred=predict()accuracy=accuracy_score(y_test,y_pred)#计算准确率print(f'y_test:{y_test}')print(f'y_pred:{y_pred}')print("准确率:",accuracy)5.2使用sklearn中的方法5.2.1数据集:鸢尾花iris(来自sklearn提供的数据集)#!高斯朴素贝叶斯分类器_鸢尾花分类importpandasaspdfromsklearn.datasetsimportload_irisfromsklearn.naive_bayesimportGaussianNBfromsklearn.model_selectionimporttrain_test_splitfromsklearn.metricsimportaccuracy_score#?定义朴素贝叶斯分类器classbayes_model():def__int__(self):pass#?加载数据defload_data(self):data=load_iris()iris_data=data.datairis_target=data.targetiris_featureNames=data.feature_namesiris_features=pd.DataFrame(data=iris_data,columns=iris_featureNames)train_x,test_x,train_y,test_y=train_test_split(iris_features,iris_target,test_size=0.3,random_state=123)returntrain_x,test_x,train_y,test_y#?训练高斯朴素贝叶斯模型deftrain_model(self,train_x,train_y):clf=GaussianNB()clf.fit(train_x,train_y)returnclf#?处理预测的数据defproba_data(self,clf,test_x,test_y):y_predict=clf.predict(test_x)#返回待预测样本的预测结果(所属类别)y_proba=clf.predict_proba(test_x)#返回预测样本属于各标签的概率accuracy=accuracy_score(test_y,y_predict)*100#计算predict预测的准确率returny_predict,y_proba,accuracy#?训练数据defexc_p(self):train_x,test_x,train_y,test_y=self.load_data()#加载数据clf=self.train_model(train_x,train_y)#训练高斯朴素贝叶斯模型clfy_predict,y_proba,accuracy=self.proba_data(clf,test_x,test_y)#利用训练好的模型clf对测试集test_x进行结果预测分析returntrain_x,test_x,train_y,test_y,y_predict,y_proba,accuracyif__name__=='__main__':train_x,test_x,train_y,test_y,y_predict,y_proba,accuracy=bayes_model().exc_p()#训练集与其标签df1df1_1=pd.DataFrame(train_x).reset_index(drop=True)df1_2=pd.DataFrame(train_y)df1=pd.merge(df1_1,df1_2,left_index=True,right_index=True)df1.columns=['sepallength(cm)','sepalwidth(cm)','petallength(cm)','petalwidth(cm)','trainclassify']print("=============================================训练集==============================================")print(f'ThetraindataSetis:\n{df1}\n')#测试集与其标签df2df2_1=pd.DataFrame(test_x).reset_index(drop=True)df2_2=pd.DataFrame(test_y)df2=pd.merge(df2_1,df2_2,left_index=True,right_index=True)df2.columns=['sepallength(cm)','sepalwidth(cm)','petallength(cm)','petalwidth(cm)','testclassify']print("=============================================测试集==============================================")print(f'ThetestdataSetis:\n{df2}\n')#预测结果tot1=pd.DataFrame([test_y,y_predict]).Ttot2=pd.DataFrame(y_proba).applymap(lambdax:'%.2f'%x)tot=pd.merge(tot1,tot2,left_index=True,right_index=True)tot.columns=['y_true','y_predict','predict_0','predict_1','predict_2']print("============================================预测结果==============================================")print('Theresultofpredictis:\n',tot)print("=============================================准确率==============================================")print(f'TheaccuracyofTestsetis:{accuracy:.2f}%')
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 会员注册

本版积分规则

QQ|手机版|心飞设计-版权所有:微度网络信息技术服务中心 ( 鲁ICP备17032091号-12 )|网站地图

GMT+8, 2025-1-13 06:35 , Processed in 0.590563 second(s), 25 queries .

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

快速回复 返回顶部 返回列表