(nd表示d中的总词条数目),因为很多词项对分类是没有价值的,比如一些停用词“的,是,在”在每个类别中都会出现,这个词项还会模糊分类的决策面,关于特征词的选取,我的这篇文章有介绍。用特征词项表示文档后,计算文档d的类别转化为:注意P(Ck|d)只是正比于后面那部分公式,完整的计算还有一个分母,但我们前面讨论了,对每个类别而已分母都是一样的,于是在我们只需要计算分子就能够进行分类了。实际的计算过程中,多个概率值P(tj|ck)的连乘很容易下溢出为0,因此转化为对数计算,连乘就变成了累加:
我们只需要从训练数据集中,计算每一个类别的出现概率P(ck)和每一个类别中各个特征词项的概率P(tj|ck),而这些概率值的计算都采用最大似然估计,说到底就是统计每个词在各个类别中出现的次数和各个类别的文档的数目:
.;
.
其中,Nck表示训练集中ck类文档的数目,N训练集中文档总数;Tjk表示词项tj在类别ck中出现的次数,V是所有类别的词项集合。这里对词的位置作了性假设,即两个词只要它们出现的次数一样,那不管它们在文档的出现位置,它们大概率值P(tj|ck)都是一样,这个位置性假设与现实很不相符,比如“放马屁”跟“马放屁”表述的是不同的内容,但实践发现,位置性假设得到的模型准确率并不低,因为大多数文本分类都是靠词的差异来区分,而不是词的位置,如果考虑词的位置,那么问题将表达相当复杂,以至于我们无从下手。
然后需要注意的一个问题是ti可能没有出现在ck类别的训练集,却出现在ck类别的测试集合中,这样因为Tik为0,导致连乘概率值都为0,其他特征词出现得再多,该文档也不会被分到ck类别,而且在对数累加的情况下,0值导致计算错误,处理这种问题的方法是采样加1平滑,即认为每个词在各个类别中都至少出现过一次,即
下面这个例子来自于参考文献1,假设有如下的训练集合测试集:
.;
.
现在要计算docID为5的测试文档是否属于China类别,首先计算个各类的概率,P(c=China)=3/4,P(c!=China)=1/4,然后计算各个类中词项的概率:
注意分母(8+6)中8表示China类的词项出现的总次数是8,+6表示平滑,6是总词项的个数,然后计算测试文档属于各个类别的概率:
可以看出该测试文档应该属于CHina类别。
文本分类实践
我找了搜狗的搜狐新闻数据的历史简洁版,总共包括汽车、财经、it、健康等9类新闻,一共162条新闻,搜狗给的数据是每一篇新闻用一个txt文件保存,我预处理了一下,把所有的新闻文档保存在一个文本文件中,每一行是一篇新闻,同时保留新闻的id,id的首字母表示类标,预处理并分词后的示例如下:
.;
.
我用62条新闻作为训练集,剩余1万条用于测试,采用互信息进行文本特征的提取,总共提取的特征词是700个左右。
分类的结果如下:
83431 10000 0.8343
总共10000条新闻,分类正确的8343条,正确率0.8343,这里主要是演示贝叶斯的分类过程,只考虑了正确率也没有考虑其他评价指标,也没有进行优化。贝叶斯分类的效率高,训练时,只需要扫描一遍训练集,记录每个词出现的次数,以及各类文档出现的次数,测试时也只需要扫描一次测试集,从运行效率这个角度而言,朴素贝叶斯的效率是最高的,而准确率也能达到一个理想的效果。
我的实现代码如下:
#!encoding=utf-81 import2 random import3 sys import4 math import5 collections import6 sys def7 shuffle():
8 '''将原来的文本打乱顺序,用于得到训练集和测试集'''
datas 9 = [line.strip() for line in sys.stdin] 10 random.shuffle(datas) 11 for line in datas: 12 print line
.;
.
13
14
15 lables = ['A','B','C','D','E','F','G','H','I'] 16 def lable2id(lable):
17 for i in xrange(len(lables)): 18 if lable == lables[i]: 19 return i
20 raise Exception('Error lable %s' % (lable)) 21 22 def docdict():
23 return [0]*len(lables) 24
25 def mutalInfo(N,Nij,Ni_,N_j): 26 #print N,Nij,Ni_,N_j
27 return Nij * 1.0 / N * math.log(N * (Nij+1)*1.0/(Ni_*N_j))/ math.log(2) 28
29 def countForMI():
30 '''基于统计每个词在每个类别出现的次数,以及每类的文档数''' 31 docCount = [0] * len(lables)#每个类的词数目 32 wordCount = collections.defaultdict(docdict) 33 for line in sys.stdin:
34 lable,text = line.strip().split(' ',1) 35 index = lable2id(lable[0]) 36 words = text.split(' ') 37 for word in words:
38 wordCount[word][index] += 1 39 docCount[index] += 1 40
41 miDict = collections.defaultdict(docdict)#互信息值 42 N = sum(docCount)
43 for k,vs in wordCount.items(): 44 for i in xrange(len(vs)): 45 N11 = vs[i]
46 N10 = sum(vs) - N11 47 N01 = docCount[i] - N11 48 N00 = N - N11 - N10 - N01
49 mi = mutalInfo(N,N11,N10+N11,N01+N11) +
50 mutalInfo(N,N10,N10+N11,N00+N10)+ mutalInfo(N,N01,N01+N11,N01+N00)+ 51 mutalInfo(N,N00,N00+N10,N00+N01) 52 miDict[k][i] = mi 53 fWords = set()
54 for i in xrange(len(docCount)): 55 keyf = lambda x:x[1][i]
56 sortedDict = sorted(miDict.items(),key=keyf,reverse=True)
.;
.
57 for j in xrange(100):
58 fWords.add(sortedDict[j][0]) 59 print docCount#打印各个类的文档数目 60 for fword in fWords: 61 print fword 62 63
def loadFeatureWord(): 65 '''导入特征词'''
66 f = open('feature.txt') 67 docCounts = eval(f.readline()) 68 features = set() 69 for line in f:
70 features.add(line.strip()) 71 f.close()
72 return docCounts,features 73 74 def trainBayes():
75 '''训练贝叶斯模型,实际上计算每个类中特征词的出现次数''' 76 docCounts,features = loadFeatureWord()
77 wordCount = collections.defaultdict(docdict) 78 tCount = [0]*len(docCounts)#每类文档特征词出现的次数 79 for line in sys.stdin:
80 lable,text = line.strip().split(' ',1) 81 index = lable2id(lable[0]) 82 words = text.split(' ') 83 for word in words: 84 if word in features: 85 tCount[index] += 1
86 wordCount[word][index] += 1 87 for k,v in wordCount.items():
88 scores = [(v[i]+1) * 1.0 / (tCount[i]+len(wordCount)) for i in xrange(len(v))]# 加1平滑
90 print '%s\%s' % (k,scores) 91 92 def loadModel(): 93 '''导入贝叶斯模型''' 94 f = open('model.txt') 95 scores = {} 96 for line in f:
97 word,counts = line.strip().rsplit('\',1) 98 scores[word] = eval(counts) 99 f.close() 100 return scores
.;
.
101
102 def predict():
103 '''预测文档的类标,标准输入每一行为一个文档''' 104 docCounts,features = loadFeatureWord()
105 docscores = [math.log(count * 1.0 /sum(docCounts)) for count in docCounts] 106 scores = loadModel() 107 rCount = 0 108 docCount = 0
109 for line in sys.stdin:
110 lable,text = line.strip().split(' ',1) 111 index = lable2id(lable[0]) 112 words = text.split(' ') 113 preValues = list(docscores) 114 for word in words:
115 if word in features: 116 for i in xrange(len(preValues)):
117 preValues[i]+=math.log(scores[word][i]) 118 m = max(preValues)
119 pIndex = preValues.index(m) 120 if pIndex == index: 121 rCount += 1
122 print lable,lables[pIndex],text 123 docCount += 1
124 print rCount,docCount,rCount * 1.0 / docCount 125 126
127 if __name__==\"__main__\": 128 #shuffle() #countForMI() #trainBayes() predict()
代码里面,计算特征词与训练模型、测试是分开的,需要修改main方法,比如计算特征词:
$cat train.txt | python bayes.py > feature.txt 1
训练模型:
$cat train.txt | python bayes.py > model.txt 1
预测模型:
.;
.
$cat test.txt | python bayes.py > predict.out 1
总结
本文介绍了朴素贝叶斯分类方法,还以文本分类为例,给出了一个具体应用的例子,朴素贝叶斯的朴素体现在条件变量之间的性假设,应用到文本分类上,作了两个假设,一是各个特征词对分类的影响是的,另一个是词项在文档中的顺序是无关紧要的。朴素贝叶斯的性假设在实际中并不成立,但在分类效上依然不错,加上性假设后,对与属于类ck的谋篇文档d,其p(ck|d)往往会估计过高,即本来预期p(ck|d)=0.55,而朴素贝叶斯却计算得到p(ck|d)=0.99,但这并不影响分类结果,这是朴素贝叶斯分类器在文本分类上效果优于预期的原因。
.;