Task7 卷积神经网络

任务:

  1. 卷积运算的定义、动机(稀疏权重、参数共享、等变表示)。一维卷积运算和二维卷积运算。
  2. 反卷积(tf.nn.conv2d_transpose)
  3. 池化运算的定义、种类(最大池化、平均池化等)、动机。
  4. Text-CNN的原理。
  5. 利用Text-CNN模型来进行文本分类。

文本分类实现数据集处理流程图

  • 数据处理步骤+每一步处理的目的如下图所示:


    文本分类实现数据集处理流程图
  • 数据处理部分实现
  1. 读数据
    其中数据是已经jiebe分词+去停用词后的结果
def read_file(path):
    with open(path, 'r', encoding="UTF-8") as f:
        data = []
        labels = []
        for line in f:
            if line.split('\t')[1] == '':
                continue
            data.append(line.split('\t')[0])
            labels.append(line.split('\t')[1])
    return data, labels
# 读文件
data, labels = read_file('E:/task6/merge.txt')
  1. 利用tokenizer将文字转换为数字特征
  • 将文字转换为数字特征要将整个数据集放入,这样保证训练集、验证集、测试集文字编码一致
  • 其中同时返回对应单词和数字的映射关系,以便在转化词向量过程中获取词向量列表
def get_tokenizer(data):
    tokenizer = Tokenizer(num_words=None)
    tokenizer.fit_on_texts(data)
    text_seq = tokenizer.texts_to_sequences(data)
    # 对应的单词和数字的映射关系
    word_index = tokenizer.word_index 
    index_word = tokenizer.index_word
    return word_index, index_word, text_seq

# 利用tokenizer将文字转换为数字特征
word_index, index_word, X_train_text_seq = get_tokenizer(data)
  1. FastText Embedding
def get_fasttext_voc(data, word_index):
    '''
        利用fasttext获取词向量
    '''
    fasttext_model = FastText([data], 
                              size=FASTEXT_SIZE,         # 需要学习的嵌入大小(默认为100)
                              window=3,         # 上下文窗口大小(默认5)
                              min_count=1,      # 忽略出现次数低于此值的单词(默认值5)
                              iter=10,          # epoch(默认5)
                              min_n = 3,        # char ngram的最小长度(默认值3)
                              max_n = 6,        # char ngram的最大长度(默认值6)
                              word_ngrams = 0)  # 如果为1,使用子单词(n-grams)信息丰富单词向量。如果是0,这就相当于Word2Vec
    # 获取词向量列表
    wordEmbedding = np.zeros((len(word_index) + 1, FASTEXT_SIZE))
    for word, i in word_index.items():
        if word in fasttext_model:
            wordEmbedding[i] = fasttext_model[word]   
    return wordEmbedding

# fasttext embedding
wordEmbedding = get_fasttext_voc(data, word_index)
  1. 让每个文本长度相同
# 让每个文本长度相同
X_train_text_seq = pad_sequences(X_train_text_seq, maxlen=LEN_WORDS, padding='post', truncating='post')
  1. 划分数据集
# 划分数据集
X_train, X_test, y_train, y_test = train_test_split(X_train_text_seq, 
                                                    labels,
                                                    test_size = 0.2,
                                                    random_state=33)
  1. 对类别变量编码
def get_label_num(data):
    data = [i.replace('\n', '') for i in data]
    y_labels = list(set(data))
    le = preprocessing.LabelEncoder()
    le.fit(y_labels)
    num_labels = len(y_labels)
    data_labels = to_categorical([le.transform([x])[0] for x in data], num_labels)
    return data_labels

# 对类别变量编码
y_train_label = get_label_num(y_train)
y_test_label = get_label_num(y_test)

CNN

实现基础版CNN:

LeNet-5 是卷积神经网络的作者Yann LeCun用于MNIST识别任务提出的模型。模型很简单,就是卷积池化层的堆叠,最后加上几层全连接层。将其运用在文本分类任务中。
其中模型结构如下所示:


实现过程如下:

def model_CNN(X_train, X_test, y_train, y_test, index_word, embedding_matrix):
    '''
        模型结构:
            嵌入层:将词编码数据转换为固定尺寸的稠密向量,同时把词向量矩阵加载到Embedding层
            卷积池化层:256 * 3 * 3
            卷积池化层:128 * 3 * 3
            Dropout:0.1
            BatchNormalization: 批量标准化层,在每一个批次的数据中标准化前一层的激活项
            全连接:256,'relu'
            分类器:2, 'softmax'
    '''
    model = Sequential()
    model.add(Embedding(len(index_word) + 1,             # imput_dim: 词汇表大小,即最大整数index+1
                        FASTEXT_SIZE,                    # output_dim: 词向量的维度
                        weights=[embedding_matrix],      # 加载词向量矩阵
                        input_length=LEN_WORDS,           # input_lenth: 输入序列的长度
                        trainable=False))                # 设置trainable=False使得这个编码层不可再训练
    # filters:输出空间的维度,kernel_size: 1D 卷积窗口的长度,padding:"same" 表示填充输入以使输出具有与原始输入相同的长度
    model.add(Conv1D(256, 3, padding='same'))   
    # pool_size:最大池化的窗口大小, strides:作为缩小比例的因数
    model.add(MaxPooling1D(3, 3, padding='same'))
    model.add(Conv1D(128, 3, padding='same'))
    model.add(MaxPooling1D(3, 3, padding='same'))
    model.add(Conv1D(64, 3, padding='same'))
    
    model.add(Flatten())
    # rate: 在 0 和 1 之间浮动。需要丢弃的输入比例
    model.add(Dropout(0.1))
    model.add(BatchNormalization())
    # units: 正整数,输出空间维度
    model.add(Dense(256, activation='relu'))
    model.add(Dropout(0.1))
    model.add(Dense(2, activation='softmax'))
    # 配置训练模型
    model.compile(loss='categorical_crossentropy',          # 表示目标应该是分类格式的
                  optimizer='adam',                         # 随机优化的一种方法
                  metrics=['accuracy']                      # 模型评估标准
                  )
    # 给定数量的迭代训练模型
    model.summary()
    model.fit(X_train, y_train, 
              batch_size=32, 
              epochs=15, 
              validation_data=(X_test, y_test))  

Text-CNN

  1. 原理

Yoon Kim在2014年 “Convolutional Neural Networks for Sentence Classification” 论文中提出TextCNN(利用卷积神经网络对文本进行分类的算法)论文翻译版,其中网络结构如下所示:


假设我们有一些句子需要对其进行分类。句子中每个词是由n维词向量组成的,也就是说输入矩阵大小为m*n,其中m为句子长度。CNN需要对输入样本进行卷积操作,对于文本数据,filter不再横向滑动,仅仅是向下移动,有点类似于N-gram在提取词与词间的局部相关性。图中共有三种步长策略,分别是2,3,4,每个步长都有两个filter(实际训练时filter数量会很多)。在不同词窗上应用不同filter,最终得到6个卷积后的向量。然后对每一个向量进行最大化池化操作并拼接各个池化值,最终得到这个句子的特征表示,将这个句子向量丢给分类器进行分类,至此完成整个流程。
其中每个层的作用,参照:https://blog.csdn.net/asialee_bird/article/details/88813385

模型结构如下所示:


TextCNN网络结构

实现过程如下:

def model_TextCNN(X_train, X_test, y_train, y_test, index_word, embedding_matrix):
    '''
        模型结构:
            词嵌入,
            卷积池化 * 3:256 * 3 * 4
            拼接三个模型的输出向量,
            全连接,
            Dropout,
            全连接
    '''
    # shape: 一个尺寸元组(整数)表明期望的输入是按批次的LEN_WORDS维向量
    main_input = Input(shape=(LEN_WORDS, ), dtype='float32')
    embed = Embedding(len(index_word) + 1,             
                        FASTEXT_SIZE,                    
                        weights=[embedding_matrix],      
                        input_length=LEN_WORDS,           
                        trainable=False)(main_input)
    # 词窗大小分别为3,4,5
    # strides指明卷积的步长
    cnn1 =  Conv1D(256, 3, padding='same', strides=1, activation='relu')(embed)
    cnn1 = MaxPooling1D(pool_size=4)(cnn1)
    cnn2 =  Conv1D(256, 4, padding='same', strides=1, activation='relu')(embed)
    cnn2 = MaxPooling1D(pool_size=4)(cnn2)
    cnn3 =  Conv1D(256, 5, padding='same', strides=1, activation='relu')(embed)
    cnn3 = MaxPooling1D(pool_size=4)(cnn3)  
    # 合并三个模型的输出向量
    cnn = concatenate([cnn1,cnn2,cnn3], axis=-1)                 
    flat = Flatten()(cnn)
    drop = Dropout(0.1)(flat)
    main_output = Dense(2, activation='softmax')(drop)
    model = Model(inputs=main_input, output=main_output)
    # 配置训练模型
    model.compile(loss='categorical_crossentropy',          # 表示目标应该是分类格式的
                  optimizer='adam',                         # 随机优化的一种方法
                  metrics=['accuracy']                      # 模型评估标准
                  )
    # 给定数量的迭代训练模型
    model.summary()
    model.fit(X_train, y_train, 
              batch_size=32, 
              epochs=15, 
              validation_data=(X_test, y_test))

完整代码见:github
参考资料:

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,128评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,316评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,737评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,283评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,384评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,458评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,467评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,251评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,688评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,980评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,155评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,818评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,492评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,142评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,382评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,020评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,044评论 2 352

推荐阅读更多精彩内容