zoukankan      html  css  js  c++  java
  • Bert模型实现垃圾邮件分类

    近日,对近些年在NLP领域很火的BERT模型进行了学习,并进行实践。今天在这里做一下笔记。

    本篇博客包含下列内容:

    BERT模型简介

      概览

      BERT模型结构

    BERT项目学习及代码走读

      项目基本特性介绍

      代码走读&要点归纳

    基于BERT模型实现垃圾邮件分类

      TREC06语料库

      基准模型介绍

      BERT迁移模型实现

    一.BERT模型简介

    1.概览

      BERT模型的全称是Bidirectional Encoder Representations from Transformer,即Transformer模型的双向编码器。只看名称可能很难看出门道,简单点讲,BERT模型就是一个Word2Vec的进化版,使用词向量对自然语言进行表示,但其模型深度极大,参数也特别的多。以Bert_BASE模型来举例,其包含12个隐藏层,每个隐层维度为768,每层又包含12个attention head,总共有110M个参数,模型参数文件在硬盘上就占据400MB的空间。

    BERT是一个预训练模型,即通过半监督学习的方式,在海量的语料库上学习出单词的良好特征表示。其在11个经典NLP任务中都展现出了最佳的性能。Bert模型一共有4个特征:

      ①预训练:是一个预先训练好的语言模型,所有未来的开发者都可以直接继承使用。

      ②深度:是一个很深的模型,Bert_BASE的层数是12,Bert_LARGE的层数是24。

      ③双向Transformer:BERT是在基于Attention原理的Transformer模型上发展而来,通过丢弃 Transformer 中的 Decoder 模块(仅保留Encoder),BERT 具有双向编码能力和强大的特征提取能力。

      ④自然语言理解:其半监督学习方式,更强调模型对自然语言的理解能力,而不是语言生成。

    2.BERT模型结构

       BERT模型的结构图如上所示。以Bert_BASE模型为例:其输入为符合化之后的向量,通过Embedding(嵌入)层,完成一些基本的预处理工作,之后就是由12个隐藏层组成的Transformer模型结构,最后的Pooling(池化)层,完成降维,输出最终结果。

    预训练工作:

      Bert模型的预训练工作包含2个任务:即掩码语言模型任务和句子对匹配检验任务。

          掩码语言模型任务:在训练过程中,从输入句子中屏蔽一些词,然后根据上下文来尝试将这些词进行复原(类似于英语考试的完形填空)。在半监督学习的过程当中,会有15%的词汇被随机屏蔽,其中的80%直接替换为[MASK],10%替换成其他词语,另外10%保持原词汇不变。

          句子对匹配检验任务:句子A和句子B一起输入到BERT模型,由BERT模型来判断句子 B 是否是后面的句子 A(True/False)。训练数据是从平行语料中随机抽取两个连续的句子生成的,50%的样本保留抽取的两个句子(True),其余50%样本的第二个句子从语料库中随机抽取(False) .

          上述2个任务都是使用维基百科作为训练语料库。

    二.BERT项目学习及代码走读

    1.项目基本特性介绍

      这里以GitHub上的bert-master项目为例(https://github.com/google-research/bert),对BERT模型的源代码进行学习,了解其流程,掌握要点。

      bert项目是Google Research最早开源的一个BERT模型项目。第一眼看过去,这个项目还真的挺复杂的,没有直接能运行的demo不说(只有代码,没有示例数据),预训练基准模型也需要另外下载。这里我下载了两个预训练模型存放在bert-master项目中的models文件夹内。基准模型下载地址可以在README.md文件中找到。

    英文BERT模型:uncased_L-12_H-768_A-12

    https://storage.googleapis.com/bert_models/2018_10_18/uncased_L-12_H-768_A-12.zip

    中文BERT模型:chinese_L-12_H-768_A-12

    https://storage.googleapis.com/bert_models/2018_11_03/chinese_L-12_H-768_A-12.zip

      通过阅读README文件,发现BERT模型的main函数在run_classifier.py代码文件中,但其使用也较为复杂,需要在运行run_classifier.py代码时传入大量的参数,README文件中示例如下:

    为了简单使用,尽量少的输入参数,制作shell脚本/bat文件,来方便的执行代码。不再对 BERT_BASE_DIR 和 GLUE_DIR 环境变量进行设置,直接将对应的数据位置、基准模型位置的相对路径进行填入即可。

    run.sh

    run.bat

    2.代码走读&要点归纳

    run_classifier.py

    1.样本类

      包含InputExample()类和PaddingInputExample()类。规定了BERT模型输入样本的格式和内容,其中PaddingInputExample()类是在样本数量不足的情况下,通常是训练的最后一个batch,填入多个空样本,将样本数量填充至batch_size。

    2.数据预处理类

          包含DataProcessor()类以及继承该类的各个子类。DataProcessor()类不提供具体的数据预处理方法,需要各子类来编写完成具体的数据预处理方法。代码中自带Xnliprocessor(),MnliProcessor(),MrpcProcessor(),ColaProcessor()四个数据预处理子类,分别对应4种不同的公共数据集。仿照这4种数据预处理子类,我们可以编写自己的数据预处理类,来对自己的数据集进行处理,以适配BERT模型,这些内容会在第三部分具体讲解。

    3. convert_single_example()函数

      将一个样本(字符串类型),经过符号化等一系列操作,转换成神经网络可以使用的InputFeature(List向量类型)。其中InputFeature包含5部分内容:

      input_ids: 即各word的位置序列(在词汇表中的位置)   如[101,123,4342,5423,632,732,....,0]

      input_mask: 即word掩码,1为真货,0为序列填充   如[1,1,1,1,1,1,1,...,0]

      segment_ids: 即2个句子的标注序列,  如[0,0,0,0,0,0,1,1,1,1,1,1]。对于 segment_ids 如2个句子后还不到最大长度,则后面用0填充。

      label_id: 即标签的位置,在 label_list 中的位置。

      is_real_example: 是否为真正样本的标记,取值为布尔值。

      该样本转换函数主要通过tokenization.py代码中的FullTokenizer()类来完成上述功能。

    4. file_based_convert_examples_to_features()函数

           将一组样本(由InputFeatures类组成),转化为 tf.train.Example类,并将转换好的样本内容写入文件output_file,包含train.tf_record,eval.tf_record,predict.tf_record 三类文件,与运行程序时的do_train、do_eval、do_predict参数相关联。需要调用上方的convert_single_example()函数。

    5. file_based_input_fn_builder()函数

          该函数是一个样本迭代器构造函数,最终返回input_fn()函数,该input_fn函数类似一个迭代器,从 train.tf_record / eval.tf_record / predict.tf_record文件中读取数据,按照 batch_size 进行数据的 shuffle(乱序),之后转数据类型为tf.int32,再返回转换后数据。

    6. model_fn_builder()函数

          该函数是一个模型函数的生成函数,其返回内容model_fn会作为参数传入tf.contrib.tpu.TPUEstimator类中,在初始化该类时加以使用。model_fn_builder()函数的主体内容即model_fn,它通过调用create_model函数,使用modeling.py代码中的BertModel类来创建深度学习网络模型,并完成加载基准BERT模型(init_checkpoint参数),定义optimizer,设置网络训练步骤等操作,最终返回的output_spec为tf.contrib.tpu.TPUEstimatorSpec类对象。

    tokenization.py

    1. FullTokenizer()类:

      该类为符号化类,在run_classifier.py中的convert_single_example()方法中加以使用。主要功能由tokenize()函数进行实现,将字符串转换为由词汇表(vocab.txt)中所包含词汇组成的List。核心为两层嵌套的for循环。外层for循环分割单词,汉字,标点符号,组成一个list;内层for循环尝试将词汇表中不存在的单词分割成多个子串,例如:”unaffable”→[“un”, ”##aff”, ”##able”],可以在保留词根的同时,扩充了词汇的延展性。实在分割不出来的未知词汇用[UNK]代替。

    2.符号化支持类:

      包括BasicTokenizer类和WordpieceTokenizer类,这两个类负责完成FullTokenizer()类中的具体功能。其中,BasicTokenizer类完成的工作包括去除口音词汇,去除标点符号,判断中文字符,去除间隔符等。WordpieceTokenizer类主要完成最大子字符串搜索功能。

    modeling.py

      在这个代码文件中,BertModel类来完成整个模型的创建,架构工作。其结构分为嵌入层,编码层(隐藏层),池化层。其这几个层是并列关系,命名空间结构关系如下图:

    其中,嵌入层的embedding_lookup ()函数,输出最基本的词向量 embeddings,其shape 为 [batch_size,seq_length,hidden_size];编码层embedding_postprocessor 函数,将三个embeddings进行加和,最终输出shape 为[batch_size,seq_length,hidden_size]的tensor;池化层在进行降维操作时,仅取seq_length 维(即第二个维度)的第一个embedding(即第一个token),池化后, 作为输出的变量self.pooled_output的shape为[batch_size,hidden_size]。

          需要特别注意的是,transformer_model()函数实现了上文中提到的attention模型。其源论文” Attention is All You Need”可以在https://arxiv.org/abs/1706.03762查看。

    三.基于Bert模型实现垃圾邮件分类

      在上一期的博客中,使用SVM来完成垃圾邮件分类工作,本期使用BERT来构造迁移模型,以实现垃圾邮件的分类。

    1.TREC06语料库

      本次实践工作依然使用2006 TREC Public Spam Corpora 语料库,包含2组数据集,即中文数据集trec06c和英文数据集trec06p。这篇博客中以中文数据集trec06c数据集作范例。

      在该数据集中,每个邮件以GBK编码单独存储在一个文件内,保存了原始邮件的所有数据,包括发送方邮箱、接收方邮箱地址、邮件发送时间等。邮件示例如下:

    数据集中共包含21766个正样本,42854个负样本,我们根据其样本索引文件(full/index)进行预处理工作,使其正负样本达到1:1的均衡比例,最终得到的索引文件中包含21766个正样本以及21766个负样本。

    2.基准模型介绍

      这里使用的基准模型是chinese_L-12_H-768_A-12模型,其具体配置参数信息如下:

    attention_probs_dropout_prob

    0.1

    directionality

    “bidi”

    hidden_act

    "gelu"

    hidden_dropout_prob

    0.1

    hidden_size

    768

    initializer_range

    0.02

    intermediate_size

    3072

    max_position_embeddings

    512

    num_attention_heads

    12

    num_hidden_layers

    12

    pooler_fc_size

    768

    pooler_num_attention_heads

    12

    pooler_num_fc_layers

    3

    pooler_size_per_head

    128

    pooler_type

    "first_token_transform"

    type_vocab_size

    2

    vocab_size

    21128

      可以看到,BERT模型的词汇表数量是21128,相比上篇博客中的SVM模型(词汇数量95963)要少很多。其原因是Bert模型以单个汉字为基础单位,而SVM模型是以词汇(词组)为基础单位。

    3.BERT迁移模型实现

             BERT模型并不能够直接判断邮件是否为垃圾邮件,其模型输出也是一个长度为hidden_size的词向量。因此需要使用BERT作为基础模型,然后加入相应的全连接层和激活函数,完成迁移模型,以对邮件进行分类。基于BERT预训练模型已具备自然语言理解能力的情况下,对模型参数进行训练和微调后,即可实现对垃圾邮件进行分类的功能。

             使用Bert-master项目制作迁移模型,主要的工作是需要将自己的trec06c数据转换成Bert模型所需要的格式。通过前面对bert-master项目代码进行阅读,发现其数据预处理部分位于run_classifier.py代码中的DataProcessor()类附近,DataProcessor()类为其父类,我们需要编写一个数据预处理子类来对trec06c数据进行处理、转换。数据预处理、转换的流程图如下:

    首先读取索引文件,获得索引列表;第二步根据索引列表,读取每一个邮件文件;对每一封邮件进行处理,包括移除头部信息获取正文、字符解码、字符串拼接等操作,将一个邮件文件转换成为一个字符串,并形成Content_List;第四步是对每一个字符串文本内容进行封装,将其转换为InputExample类的对象,并形成Example_List,这是所有样本的集合;最后根据8:1:1的比例,划分训练集(train set)、验证集(dev set)和测试集(test set)。根据上述流程完成CN_trec06c_Processor(DataProcessor)类,其代码如下:

      1 # TODO ===============自己的dataProcessor trec06c 数据 原始email,散装数据=====================
      2 # 需要继承 DataProcessor,并重新里面的几个数据预处理函数。
      3 class CN_trec06c_Processor(DataProcessor):    
      4   def __init__(self):
      5     # 定义一些超参数
      6     self.MAX_EMAIL_LENGTH = 400   #最长单个邮件长度
      7   def get_email_file(self,base_path,path_list):
      8     email_str_list = []
      9     for i in range(len(path_list)):
     10       with open(base_path + path_list[i][1:],'r',encoding='gbk') as fin:
     11         words = ""
     12         begin_tag = 0
     13         wrong_tag = 0
     14         while(True):
     15           if wrong_tag > 20 or len(words)>self.MAX_EMAIL_LENGTH:
     16             break
     17           try:
     18             line = fin.readline()
     19             wrong_tag = 0
     20           except:
     21             wrong_tag += 1
     22             continue
     23           if (not line):
     24             break
     25           if(begin_tag == 0):
     26             if(line=='
    '):
     27               begin_tag = 1
     28             continue
     29           else:
     30             words += line.strip() + ' '
     31             if len(words)>self.MAX_EMAIL_LENGTH:
     32               break
     33         if len(words)>=10:   # 语句最短长度
     34           email_str_list.append(words)
     35     return email_str_list
     36   def get_all_examples(self):
     37     trec06Path = "../../02_SVM_analysis/data/"  # trec06c数据位置
     38     path_list_spam = []
     39     with open(trec06Path+'CN_index_spam','r',encoding='utf-8') as fin:
     40       for line in fin.readlines():
     41         path_list_spam.append(line.strip())
     42     path_list_ham = []
     43     with open(trec06Path+'CN_index_ham','r',encoding='utf-8') as fin:
     44       for line in fin.readlines():
     45         path_list_ham.append(line.strip())
     46     # 是否对原始数据长度作裁剪  共 21766 个 正例  21766个 负例
     47     path_list_spam = path_list_spam[:100]  #这里仅取100个样本进行本机测试
     48     path_list_ham = path_list_ham[:100]
     49     spam_email_list = self.get_email_file(trec06Path[:-6],path_list_spam)
     50     ham_email_list = self.get_email_file(trec06Path[:-6],path_list_ham)
     51     print("*****************====================*****************")
     52     print("正例样本数量: ",len(ham_email_list))
     53     print("反例样本数量: ",len(spam_email_list))
     54     with open('model_spam_tuning/CN_trec06c/train_sample_stat.txt','w',encoding='utf-8') as fout:
     55       fout.write("正例样本数量: " + str(len(ham_email_list)) + '
    ')
     56       fout.write("反例样本数量: " + str(len(spam_email_list)) + '
    ')
     57     print("*****************====================*****************")
     58     examples = []
     59     for i in range(len(spam_email_list)):
     60       guid = "train-%d" % (i)  # 从 0 开始
     61       # TODO 下方,tokenization.convert_to_unicode() 函数,将byte类数据 decode成为'utf-8'
     62       text_a = tokenization.convert_to_unicode(str(spam_email_list[i]))
     63       label = '0' # 转int类 是后续的操作,此处仍旧是str 垃圾邮件label为0
     64       examples.append(
     65           InputExample(guid=guid, text_a=text_a, text_b=None, label=label))
     66     for i in range(len(ham_email_list)):
     67       guid = "train-%d" % (i+len(spam_email_list))  # 从 垃圾邮件长度 开始向后续
     68       # TODO 下方,tokenization.convert_to_unicode() 函数,将byte类数据 decode成为'utf-8'
     69       text_a = tokenization.convert_to_unicode(str(ham_email_list[i]))
     70       label = '1' # 转int类 是后续的操作,此处仍旧是str 正常邮件label为1
     71       examples.append(
     72           InputExample(guid=guid, text_a=text_a, text_b=None, label=label))
     73     # TODO 还要进行切分 测试集,训练集,验证集
     74     return examples
     75 
     76   def get_train_examples(self, data_dir):
     77     # train_data_path = os.path.join(data_dir, "cn_train_tiny_tiny.csv") # 训练集 数据文件名称,可以在这里改
     78     examples = self.get_all_examples()
     79     ex_new = []
     80     for i in range(len(examples)):
     81       if i%10 != 1 and i%10 != 2: # 8:1:1 切分训练集,验证集,测试集
     82         ex_new.append(examples[i])
     83     return ex_new
     84 
     85   def get_dev_examples(self, data_dir):
     86     """Gets a collection of `InputExample`s for the dev set."""
     87     examples = self.get_all_examples()
     88     ex_new = []
     89     for i in range(len(examples)):
     90       if i%10 == 1:  # 8:1:1 切分训练集,验证集,测试集
     91         ex_new.append(examples[i])
     92     return ex_new
     93 
     94   def get_test_examples(self, data_dir):
     95     """Gets a collection of `InputExample`s for prediction."""
     96     test_set_txt = []
     97     test_set_label = []
     98     examples = self.get_all_examples()
     99     ex_new = []
    100     for i in range(len(examples)):
    101       if i%10 == 2:  # 8:1:1 切分训练集,验证集,测试集
    102         ex_new.append(examples[i])
    103         test_set_txt.append(examples[i].text_a)
    104         test_set_label.append(examples[i].label)
    105     with open('model_spam_tuning/CN_trec06c/test_origin.txt','w',encoding='utf-8') as fout:
    106       for i in range(len(test_set_label)):
    107         fout.write(test_set_label[i]+'	'+test_set_txt[i]+'
    ')
    108     return ex_new
    109 
    110   def get_labels(self):
    111     """Gets the list of labels for this data set."""
    112     return ['0','1']
    113 # TODO ===============自己的dataProcessor trec06c 数据=====================

      其中,get_email_file()函数读取路径List中所有邮件文件的内容,并完成预处理、筛选工作,返回邮件内容List;get_all_examples()函数分别读取垃圾邮件路径索引List以及正常邮件路径索引List,调用get_email_file()函数获取所有邮件的内容,将格式化为BERT模型所需要的格式,即InputExample()类,包括guid,text_a,text_b,label四个属性。后面get_train_examples(),get_dev_examples(),get_test_examples()三个函数对所有的邮件样本进行划分,分别得到训练集,验证集,测试集。需要注意的是,这三个函数的名称不能随意更动,它们是对父类DataProcessor()中同名方法的具体实现。最后一个函数get_labels()返回标签列表,分几类有返回几种标签,需要注意的是这里的标签仍然是字符串类型。

      编写完成数据预处理类之后,我们需要给Bert模型添加相应的处理任务,将CN_trec06c_Processor类添加到下方main()函数的processors字典变量中,如下图。需要注意字典的key值必须全部用小写字母。

    这时,基于BERT的迁移模型已经构建完毕,使用前面编写好的run.sh(Linux)或run.bat(windows)即可运行run_classifier.py脚本开始训练。由于bert-master项目会将所有的数据样本读入显存进行训练,因此在我本机环境下难以运行(GTX 860、2G显存),仅能运行100个样本。因此,借用一个朋友的云服务器(GTX 1080Ti、10G显存)开展实验。硬件配置&环境配置版本如下:

    在超参数配置为下表的情况下,训练了25个epochs,耗时约3个小时。

    下图为训练时的输出,可以看出云服务器每秒可以完成120多个样本的训练。

     最终得到的结果在模型文件夹的eval_result.txt文件中,最终模型在验证集上的分类准确率达到了99.347%,比之前的SVM模型要准确不少。

     而该BERT迁移模型对于测试集的分类结果则在predict_results.tsv文件中。

      对于英文邮件建立BERT迁移模型的方法与中文类似,需要完成一个英文邮件的数据预处理类,并将其添加到main()函数的processors字典变量中。代码就不罗列了,最终得到的英文垃圾邮件分类准确率为98.81% 。

     
  • 相关阅读:
    PM对功能模块的说明
    第四个迭代目标
    第三个迭代任务
    第三个迭代目标
    记账本状态图
    记账本数据流图
    记账本ER图
    记账本用例图
    第二个迭代目标
    记账本时序图
  • 原文地址:https://www.cnblogs.com/NosenLiu/p/14956415.html
Copyright © 2011-2022 走看看