文章目录

  • 利用朴素贝叶斯进行文档分类
    • 1、获取数据集
    • 2、切分文本
    • 3、构建词表和分类
    • 4、构建分类器
    • 5、测试算法
  • 利用朴素贝叶斯进行垃圾邮件过滤
    • 1、导入数据集
    • 2、垃圾邮件预测
  • 总结

利用朴素贝叶斯进行文档分类

1、获取数据集

下载数据集,获取到一些邮件文档。其中ham文件夹下都是正常邮件,spam下都是垃圾邮件。

数据集中将一些词组用空格分开,方便后面分割词组,统计词表。

2、切分文本

每一份邮件里面用空格键分割出很多的词组,需要将这些词组分割出来,放在一个词表里面。

def cut_sentences(content):   # 实现分句的函数,content参数是传入的文本字符串end_flag = ['?', '!', '.', '?', '!', '。', ' ']   # 结束符号,包含中文和英文的content_len = len(content)sentences = []   # 存储每一个句子的列表tmp_char = ''for idx, char in enumerate(content):tmp_char += char   # 拼接字符if (idx + 1) == content_len:   # 判断是否已经到了最后一位sentences.append(tmp_char.strip().replace('\ufeff', ''))breakif char in end_flag:   # 判断此字符是否为结束符号# 再判断下一个字符是否为结束符号,如果不是结束符号,则切分句子next_idx = idx + 1if not content[next_idx] in end_flag:sentences.append(tmp_char.strip().replace('\ufeff', ''))tmp_char = ''sentences = list(set(sentences))return  [tok for tok in sentences if len(tok) > 1]words = cut_sentences(open('TrainingSet/Ham/{}.txt'.format(2)).read())
print(words)

例如分割好一份邮件后,可以得到一个词表

3、构建词表和分类

下面方法中,posting_list 存放着每一个邮件的词表,class_vec 里面存放对应下标词表的类别,1代表友好文件,0代表不友好文件。最后返回一个词表的列表,和一个类别的列表。

# 获取词表和对应类别
def load_data_set():# 词表和对应类别posting_list = []class_vec = []for i in range(2, 10):# 获取友好邮件词表words = cut_sentences(open('TrainingSet/Ham/{}.txt'.format(i)).read())posting_list.append(words)class_vec.append(1)# 获取不友好邮件词表words = cut_sentences(open('TrainingSet/Spam/{}.txt'.format(i)).read())posting_list.append(words)class_vec.append(0)return posting_list, class_vec


create_vocab_list方法会将load_data_set返回的词表列表合并成一个大的词表,里面每一个词组输给Set函数后,set会返回一个不重复的大词表。

# 合并所有词表
def create_vocab_list(data_set):vocab_set = set() for item in data_set:# | 求两个集合的并集vocab_set = vocab_set | set(item)return list(vocab_set)
print(create_vocab_list(posting_list))

通过上面create_vocab_list方法可以返回一个包含所有词汇的大词表,用set_of_words2vec方法,传入大词表与小词表,可以遍历出小词表中每个词汇在大词表中存在的位置,对应位置标志置为1。创建一个和词汇表等长的向量,并将其元素都设置为0,遍历文档中的所有单词,如果出现了词汇表中的单词,则将输出的文档向量中的对应值设为1。

def set_of_words2vec(vocab_list, input_set):# 创建一个和词汇表等长的向量,并将其元素都设置为0result = [0] * len(vocab_list)# 遍历文档中的所有单词,如果出现了词汇表中的单词,则将输出的文档向量中的对应值设为1for word in input_set:if word in vocab_list:result[vocab_list.index(word)] = 1else:passreturn result

4、构建分类器

利用公式:
P(ci∣w)=P(w∣ci)P(ci)P(w)P(c_{i}|\mathbf{w } ) = \frac{P(\mathbf{w }|c_{i})P(c_{i})}{P(\mathbf{w } )} P(ci​∣w)=P(w)P(w∣ci​)P(ci​)​
首先可以通过类别i(友好或者非友好)邮件数除以总的邮件数计算概率P(ci)。接着计算P(w|ci),这里就要用到贝叶斯假设。如果将w展开为一个个独立特征,那么就可以将上述概率写作P(w0,w1,w2...wn),这里假设所有词都互相独立,该假设也作为条件独立性假设,它意味着可以使用P(w0,ci)P(w1,ci)P(w2,ci)...P(wn,ci)计算概率。

     该函数的伪代码如下:计算每个类别中的文档数目对每篇训练文档:对每个类别:如果词条出现在文档中→ 增加该词条的计数值增加所有词条的计数值对每个类别:对每个词条:将该词条的数目除以总词条数目得到条件概率返回每个类别的条件概率

计算P(wi|c1)P(wi|c1),需要初始化程序中的分子变量和分母变量。一旦每个词语(友好或者不友好)在某一文件中出现,则改词对应的个数(p0Num或者p1Num)就加1,而在所有文件中,该文件总词数也对应加1。


def train_naive_bayes(train_mat, train_category):train_doc_num = len(train_mat)words_num = len(train_mat[0])# 因为侮辱性的被标记为了1, 所以只要把他们相加就可以得到侮辱性的有多少# 侮辱性文件的出现概率,即train_category中所有的1的个数,# 代表的就是多少个侮辱性文件,与文件的总数相除就得到了侮辱性文件的出现概率pos_abusive = np.sum(train_category) / train_doc_num# 单词出现的次数p0num = np.ones(words_num)p1num = np.ones(words_num)print(p1num)p0num_all = 2.0p1num_all = 2.0for i in range(train_doc_num):# 遍历所有的文件,如果是侮辱性文件,就计算此侮辱性文件中出现的侮辱性单词的个数if train_category[i] == 1:p1num += train_mat[i]p1num_all += np.sum(train_mat[i])else:p0num += train_mat[i]p0num_all += np.sum(train_mat[i])p1vec = np.log(p1num / p1num_all)p0vec = np.log(p0num / p0num_all)return p0vec, p1vec, pos_abusive

vec2classify: 待测数据[0,1,1,1,1…],即要分类的向量,p0vec: 类别0,即正常文档的, p1vec: 类别1,即侮辱性文档的,p_class1类别1,侮辱性文件的出现概率。使用算法: 乘法: P(C|F1F2...Fn) = P(F1F2...Fn|C)P(C)/P(F1F2...Fn),加法:P(F1|C)*P(F2|C)....P(Fn|C)P(C) -> log(P(F1|C))+log(P(F2|C))+....+log(P(Fn|C))+log(P(C)),使用 NumPy 数组来计算两个向量相乘的结果,这里的相乘是指对应元素相乘,即先将两个向量中的第一个元素相乘,然后将第2个元素相乘,以此类推。我的理解是: 这里的 vec2Classify * p1Vec 的意思就是将每个词与其对应的概率相关联起来,可以理解为 1.单词在词汇表中的条件下,文件是good 类别的概率 也可以理解为 2.在整个空间下,文件既在词汇表中又是good类别的概率。

def classify_naive_bayes(vec2classify, p0vec, p1vec, p_class1):p1 = np.sum(vec2classify * p1vec) + np.log(p_class1)p0 = np.sum(vec2classify * p0vec) + np.log(1 - p_class1)if p1 > p0:return 1else:return 0

5、测试算法

输入测试文件的对应词表,可以判断这个文件是友好的还是不友好的。例如我输入两个文件一部分词表,通过算法可以判断该邮件是友好还是非友好文件。其中词表['一番', '卑劣', '某人', '冷静', '以为', '真的', '不能', '有着', '这件', '清华', '心情 。', '一起 。', '解释',]是从友好文件中截取的,['邮箱', '地税', '查证', '树立', '较多现', '广州', '我司', '大小', '如贵司', '额度', '核心思想', '电脑', '真票 !', '每月', '洽商']是从非友好文件中截取的,最后算法预测的结果也是正确的。

def testing_naive_bayes():# 1. 加载数据集list_post, list_classes = load_data_set()# 2. 创建单词集合vocab_list = create_vocab_list(list_post)# 3. 计算单词是否出现并创建数据矩阵train_mat = []for post_in in list_post:train_mat.append(# 返回m*len(vocab_list)的矩阵, 记录的都是0,1信息set_of_words2vec(vocab_list, post_in))# 4. 训练数据p0v, p1v, p_abusive = train_naive_bayes(np.array(train_mat), np.array(list_classes))# 5. 测试数据test_one = ['一番', '卑劣', '某人', '冷静', '以为', '真的', '不能', '有着', '这件', '清华', '心情 。', '一起 。', '解释',]print(test_one)test_one_doc = np.array(set_of_words2vec(vocab_list, test_one))print('这个邮件的类别是: {}'.format(classify_naive_bayes(test_one_doc, p0v, p1v, p_abusive)))test_two = ['邮箱', '地税', '查证', '树立', '较多现', '广州', '我司', '大小', '如贵司', '额度', '核心思想', '电脑', '真票 !', '每月', '洽商']print(test_two)test_two_doc = np.array(set_of_words2vec(vocab_list, test_two))print('这个邮件的类别是: {}'.format(classify_naive_bayes(test_two_doc, p0v, p1v, p_abusive)))

利用朴素贝叶斯进行垃圾邮件过滤

1、导入数据集

数据集中友好邮件和不友好邮件大概都是8500份。

2、垃圾邮件预测

垃圾邮件预测算法主要用到利用朴素贝叶斯进行文档分类中的贝叶斯分类器,通过分类器返回判断类型与实际类型比较,最终可以计算出错误比例。我这里友好文件和不友好文件都有1000份,对这些邮件处理返回概率之后,再从总共份文件中随机抽取100份进行预测,最后输出预测的结果。

def spam_test():"""对贝叶斯垃圾邮件分类器进行自动化处理。:return: nothing"""doc_list = []class_list = []full_text = []for i in range(1, 51):try:words = cut_sentences(open('TrainingSet/Ham/{}.txt'.format(i)).read())except:words = cut_sentences(open('TrainingSet/Ham/{}.txt'.format(i), encoding='Windows 1252').read())doc_list.append(words)full_text.extend(words)class_list.append(1)try:# 添加非垃圾邮件words = cut_sentences(open('TrainingSet/Ham/{}.txt'.format(i)).read())except:words = cut_sentences(open('TrainingSet/Ham/{}.txt'.format(i), encoding='Windows 1252').read())doc_list.append(words)full_text.extend(words)class_list.append(0)# 创建词汇表vocab_list = create_vocab_list(doc_list)import randomtest_set = [int(num) for num in random.sample(range(100), 50)]training_set = list(set(range(100)) - set(test_set))training_mat = []training_class = []for doc_index in training_set:training_mat.append(set_of_words2vec(vocab_list, doc_list[doc_index]))training_class.append(class_list[doc_index])p0v, p1v, p_spam = train_naive_bayes(np.array(training_mat),np.array(training_class))# 开始测试error_count = 0for doc_index in test_set:word_vec = set_of_words2vec(vocab_list, doc_list[doc_index])if classify_naive_bayes(np.array(word_vec),p0v,p1v,p_spam) != class_list[doc_index]:error_count += 1print('正确率为 {}'.format(1 - (error_count / len(test_set))))

总结

算法对大量的数据集不太友好,最开始我找了上万条的邮件作为数据集,最后运行出来的结果特别差错误率能达到0.9。最后不得不减少数据集,分别用500条友好和不友好的文件,一共抽取100条测试正确率,最后结果正确率只有0.18。

继续减小数据集,最后分别用50条友好和不友好的文件,一共抽取20条测试正确率,正确率只有0.28。应该是我的数据集里面每一封邮件内容过多,中文词组之间没有太大联系造成的。

朴素贝叶斯算法优点:

  • 朴素贝叶斯模型有稳定的分类效率。
  • 对小规模的数据表现很好,能处理多分类任务,适合增量式训练,尤其是数据量超出内存时,可以一批批的去增量训练。
  • 对缺失数据不太敏感,算法也比较简单,常用于文本分类。

缺点:

  • 理论上,朴素贝叶斯模型与其他分类方法相比具有最小的误差率。但是实际上并非总是如此,这是因为朴素贝叶斯模型给定输出类别的情况下,假设属性之间相互独立,这个假设在实际应用中往往是不成立的,在属性个数比较多或者属性之间相关性较大时,分类效果不好。而在属性相关性较小时,朴素贝叶斯性能最为良好。对于这一点,有半朴素贝叶斯之类的算法通过考虑部分关联性适度改进。
  • 需要知道先验概率,且先验概率很多时候取决于假设,假设的模型可以有很多种,因此在某些时候会由于假设的先验模型的原因导致预测效果不佳。
  • 对输入数据的表达形式很敏感。

代码地址:
链接:https://pan.baidu.com/s/1iXUTGRxISzWisOk_Wt6xXA?pwd=zrqx
提取码:zrqx

朴素贝叶斯——垃圾邮件过滤相关推荐

  1. 第四章 朴素贝叶斯-垃圾邮件过滤

    #朴素贝叶斯之垃圾邮件过滤 import os #----------------------1.获取数据集----------------------- import pandas as pd''' ...

  2. php贝叶斯,php – 将单个概率与朴素贝叶斯垃圾邮件过滤相结合

    我正在尝试通过分析我已经积累的语料库来生成垃圾邮件过滤器. 我已经实现了代码来计算消息是垃圾邮件的概率,因为它包含一个特定的单词,通过从wiki实现以下公式: 我的PHP代码 public funct ...

  3. Python实现基于朴素贝叶斯的垃圾邮件分类 标签: python朴素贝叶斯垃圾邮件分类 2016-04-20 15:09 2750人阅读 评论(1) 收藏 举报 分类: 机器学习(19) 听说

    Python实现基于朴素贝叶斯的垃圾邮件分类 标签: python朴素贝叶斯垃圾邮件分类 2016-04-20 15:09 2750人阅读 评论(1) 收藏 举报  分类: 机器学习(19)  听说朴 ...

  4. 朴素贝叶斯-垃圾邮件(中文的)处理

    #coding=utf-8 from sklearn.naive_bayes import MultinomialNB,GaussianNB import numpy as np import jie ...

  5. python实现朴素贝叶斯垃圾邮件分类

    查看通俗易懂的贝叶斯垃圾邮件分类原理 请点击此处 下载邮件数据 请点击此处 import os import re import string import math import numpy as ...

  6. 朴素贝叶斯-垃圾邮件分类实现

    1. 前言 <朴素贝叶斯算法(Naive Bayes)>,介绍了朴素贝叶斯原理.本文介绍的是朴素贝叶斯的基础实现,用来垃圾邮件分类. 2. 朴素贝叶斯基础实现 朴素贝叶斯 (naive B ...

  7. 机器学习-朴素贝叶斯-垃圾邮件

    一:朴素贝叶斯算法概述 1:朴素贝叶斯(Naïve Bayes, NB)算法,是一种基于贝叶斯定理与特征条件独立假设的分类方法.朴素:特征条件独立:贝叶斯:基于贝叶斯定理.属于监督学习的生成模型,实现 ...

  8. 机器学习实战4(1):朴素贝叶斯:垃圾邮件的识别

    一.朴素贝叶斯基础知识 预备数学知识: A. 无约束条件的优化 1.求极值问题 人工智能中最核心的数学环节是求出一个目标函数(object function)的最小值/最大值.求出一个函数最小是/最大 ...

  9. 神奇的贝叶斯---垃圾邮件过滤

    使用贝叶斯进行垃圾邮件过滤的原理很简单,假如我们在邮件中发现一个词--"发票",那么就根据这个词来计算这封邮件是垃圾邮件的概率 P(垃圾|发票) = P(发票|垃圾)P(垃圾)/P ...

最新文章

  1. 我为什么选择比特币现金
  2. [开发工具]借助dillinger.io,在博客园写markdown
  3. mysql转sqlserver_mysql转sqlserver工具
  4. 万能素材库_自媒体运营必备3款黑科技工具,一个万能素材网站,你都在用吗?...
  5. Java 中创建对象方式
  6. php获取视频大小代码怎么写,利用php获得flv视频长度的实例代码
  7. jquery的ajaxSetup()函数用法:设置全局的ajax默认选项
  8. mysql时间设计模式_java 23种设计模式及具体例子 收藏有时间慢慢看
  9. 大学四年,这个Java自习路线,让我现在进了阿里
  10. 2019 acm-icpc 西安全国邀请赛 J
  11. 笔记本电脑找不到计算机配置,笔记本电脑上的设置不见了怎么办
  12. 两个long型数据相除结果错误问题解决
  13. VM安装CentOS虚拟机详细教程
  14. 秒杀系统的设计五大原则
  15. 从成本角度理解盈利和亏损
  16. 【问题处理】Error response from daemon: Pool overlaps with other one on this address space
  17. 2021韶关市高考成绩查询,韶关高考成绩查询入口
  18. HTML京东商品页面代码,jquery仿京东商品页面代码分享
  19. CEO、CFO、COO
  20. 蔬菜视觉分拣机器人的设计与实现(RoboWork参赛方案)

热门文章

  1. SVG奥林匹克五环动画
  2. 恶搞 GitHub 贡献图
  3. bcd转ascii码 流程图_bcd码转ascii码代码
  4. 算法之地推算法(逆推法)
  5. Population and carrying capacity 的第三个阶段:Crowding affects births
  6. 360手机:360N5 Twrp、Root、Magisk教程
  7. unity之子弹发射
  8. 通过监控Nginx日志来实时屏蔽高频恶意访问的IP
  9. navicat设计表怎么给列加注释(简洁版)
  10. 前沿分享|数澜科技联合创始人副总裁 江敏:基于云原生数据仓库AnalyticDB PostgreSQL的最佳实践