原文:Implementing a CNN for Text Classification in TensorFlowhtml
做者:Denny Britznode
翻译:Kaiserpython
欢迎访问集智主站:集智,通向智能时代的引擎
git
完整代码可见做者Githubgithub
本文所部署的模型与Kim Yoon的Convolutional Neural Networks for Sentence Classification类似。论文中的模型在多种文本分类任务(好比情感分析)上都表现良好,现已加入标准基线(baseline)豪华套餐。api
我假设你已经很是熟悉卷积神经网络用于天然语言处理的那一套理论了,若是还没的话,建议先阅读前篇:一文读懂CNN如何用于NLP。bash
本文全部代码都可在集智原贴中运行调试,有须要的同窗能够点击这里前往原贴
网络
此次要用到的数据集是Movie Review data from Rotten Tomatoes——也是原论文中用到的。数据集包含10662个影评语句样本,正负面各半。数据集词汇量大约2万,注意由于数据量并不大,因此很强的模型反而容易过拟合。数据集自己没有划分出训练/测试,因此咱们简单地用10%做为测试集,原论文则是用了10折交叉验证。session
数据预处理过程以下:app
本文将要搭建的网络大体以下:
第一层将词嵌入低维向量。下一层用不一样大小的卷积核对词嵌入作卷积,每次3-5个词,而后再最大池化得到长特征向量(注:这里的“特征向量”是feature vector,不是eigenvector),通过Dropout正则化以后由softmax分类。
由于是教程文章因此我决定对原论文的模型作些简化:
固然要加上这些扩展也不难,只须要几行代码,能够看本文最后的练习。下面开始说正事:
为了便于调教超参数,咱们把代码放进名为TextCNN
的类里,用init
函数生成模型图。
import tensorflow as tf
import numpy as np
class TextCNN(object):
""" A CNN for text classification. Uses an embedding layer, followed by a convolutional, max-pooling and softmax layer. """
def __init__(self, sequence_length, num_classes, vocab_size,
embedding_size, filter_sizes, num_filters):
# Implementation...
复制代码
为了建立对象咱们须要传入如下参数:
sequence_length
:句子长度,在预处理环节咱们已经填充句子,以保持相同长度(59);num_classes
:输出层的类别数,这里是2(好评/差评);vocab_size
:词空间大小,用于定义嵌入层维度:[vocabulary_size, embedding_size];embedding_size
:嵌入维度;filter_size
:卷积核覆盖的词汇数,每种尺寸的数量由num_filters
定义,好比[3,4,5]表示咱们有3 * num_filters
个卷积核,分别每次滑过三、四、5个词。num_filters
:如上。class TextCNN(object):
def __init__(self, sequence_length, num_classes, vocab_size, embedding_size, filter_sizes, num_filters):
# Placeholders for input, output and dropout
self.input_x = tf.placeholder(tf.int32, [None, sequence_length], name="input_x")
self.input_y = tf.placeholder(tf.float32, [None, num_classes], name="input_y")
self.dropout_keep_prob = tf.placeholder(tf.float32, name="dropout_keep_prob")
复制代码
tf.placeholder
建立占位符,也就是在训练或测试时将要输入给神经网络的变量。第二个设置项是输入张量的形状。None
表示这一维度能够是任意值,使网络能够处理任意数量的批数据。
在dropout层保留一个神经元的几率也是网络的输入之一,由于咱们在训练过程当中开启了dropout,在评估和与测试这一项会停用。
网络的第一层是嵌入层,将词汇映射到低维向量表征,就像是从数据中学习到一张速查表。
class TextCNN(object):
def __init__():
...
with tf.device('/cpu:0'), tf.name_scope("embedding"):
W = tf.Variable(tf.random_uniform([vocab_size, embedding_size], -1.0, 1.0),
name="W")
self.embedded_chars = tf.nn.embedding_lookup(W, self.input_x)
self.embedded_chars_expanded = tf.expand_dims(self.embedded_chars, -1)
复制代码
注意这里的缩进,以上代码仍然是
def init():
中的一部分。
tf.device("/cpu:0")
:强制使用CPU。默认状况下TensorFlow会尝试调用GPU,可是词嵌入操做们目前尚未GPU支持,因此有可能会报错。tf.name_scope
:新建了一个命名域“embedding”,这样在TensorBoard可视化网络的时候,操做会具备更好的继承性。W
就是咱们的嵌入矩阵,也是训练中学习的目标,用随机均匀分布初始化。tf.nn.embedding_lookup
建立了实际的嵌入操做,输出结果是3D张量,形如[None, sequence_length, embedding_size]
{:tensorflow:}TensorFlow的卷积操做conv2d接收4维张量[batch, width, height, channel],而咱们的嵌入结果没有通道维,因此手动加一个变成[None, sequence_length, embedding, 1]。
如今咱们来搭建卷积层和紧随其后的池化层,注意咱们的卷积核有多种不一样的尺寸。由于每一个卷积产生的张量形状不一,咱们须要迭代对每个建立一层,而后再把结果融合到一个大特征向量里。
class TextCNN(object):
def __init__():
...
pooled_outputs = []
for i, filter_size in enumerate(filter_sizes):
with tf.name_scope("conv-maxpool-%s" % filter_size):
# Convolution Layer
filter_shape = [filter_size, embedding_size, 1, num_filters]
W = tf.Variable(tf.truncated_normal(filter_shape, stddev=0.1), name="W")
b = tf.Variable(tf.constant(0.1, shape=[num_filters]), name="b")
conv = tf.nn.conv2d(
self.embedded_chars_expanded,
W,
strides=[1, 1, 1, 1],
padding="VALID",
name="conv")
# Apply nonlinearity
h = tf.nn.relu(tf.nn.bias_add(conv, b), name="relu")
# Max-pooling over the outputs
pooled = tf.nn.max_pool(h,
ksize=[1, sequence_length - filter_size + 1, 1, 1],
strides=[1, 1, 1, 1],
padding='VALID',
name="pool")
pooled_outputs.append(pooled)
# Combine all the pooled features
num_filters_total = num_filters * len(filter_sizes)
self.h_pool = tf.concat(3, pooled_outputs)
self.h_pool_flat = tf.reshape(self.h_pool, [-1, num_filters_total])
复制代码
这里W
是卷积矩阵,h
是通过非线性激活函数的输出。每一个卷积核都在整个词嵌入空间中扫过,可是每次扫过的个数不一。"VALID" padding 表示卷积核不在边缘作填补,也就是“窄卷积”,输出形状是[1, sequence_length - filter_size + 1, 1, 1]。
最大池化让咱们的张量形状变成了[batch_size, 1, 1, num_filters],最后一位对应特征。把全部通过池化的输出张量组合成一个很长的特征向量,形如[batch_size, num_filter_total]。在TensorFlow里,若是须要将高维向量展平,能够在tf.reshape
中设置-1
。
Dropout大概是正则化卷积神经网络最流行的方法。其背后的原理很简单,就是随机“抛弃”一部分神经元,以此防止他们共同适应(co-adapting),并强制他们独立学习有用而特征。不被抛弃的比例咱们经过dropout_keep_prob
这个变量来控制,训练过程当中设为0.5,评估过程当中设为1.
class TextCNN(object):
def __init__():
...
# Add dropout
with tf.name_scope("dropout"):
self.h_drop = tf.nn.dropout(self.h_pool_flat, self.dropout_keep_prob)
复制代码
借助最大池化+dropout所得的特征向量,咱们能够作个矩阵乘法并选择分数最高的类,来作个预测。固然也能够用softmax函数来把生数据转化成正规化的几率,但这并不会改变最终的预测结果。
class TextCNN(object):
def __init__():
...
with tf.name_scope("output"):
W = tf.Variable(tf.truncated_normal([num_filters_total, num_classes], stddev=0.1), name="W")
b = tf.Variable(tf.constant(0.1, shape=[num_classes]), name="b")
self.scores = tf.nn.xw_plus_b(self.h_drop, W, b, name="scores")
self.predictions = tf.argmax(self.scores, 1, name="predictions")
复制代码
这里tf.nn.xw_plus_b
是的封装。
有了这个分数即咱们就能够定义损失函数了,损失函数(loss)是衡量网络预测偏差的指标。咱们的目标即是最小化。分类问题中标准的损失函数是交叉熵, cross-entropy loss。
class TextCNN(object):
def __init__():
...
# Calculate mean cross-entropy loss
with tf.name_scope("loss"):
losses = tf.nn.softmax_cross_entropy_with_logits(self.scores, self.input_y)
self.loss = tf.reduce_mean(losses)
复制代码
tf.nn.softmax_cross_entropy_with_logits是给定分数和正确输入标签以后,计算交叉熵的函数,而后对损失取平均值。咱们也能够求和,但那样的话不一样batch size的损失就很难比较了。
咱们也定义了精度的表达,这个量在追踪训练和测试过程当中颇有用。
class TextCNN(object):
def __init__():
...
# Calculate Accuracy
with tf.name_scope("accuracy"):
correct_predictions = tf.equal(self.predictions, tf.argmax(self.input_y, 1))
self.accuracy = tf.reduce_mean(tf.cast(correct_predictions, "float"), name="accuracy")
复制代码
至此咱们已经完成了网络的定义工做,经过TensorBoard可视化以下:
在启动训练进程以前咱们仍是有必要了解一些{:tensorflow:}TensorFlow关于“会话”(Session
)和图(Graphs
)的基本概念。若是你已经很是熟悉这一套,能够跳过本节。
TensorFlow 1.5版本已经正式推出,引入了“动态图”机制,所以本文所述并不是最新用法。
在TensorFlow当中,Session是执行计算图操做所在的环境,包含变量和队列的状态。每一个Session执行一个图,若是没有显式地调用session,那么在建立变量和操做时,就使用TF默认建立的。能够运行命令session.as_default()
更改默认session。
一个图(Graph)包含操做和张量,每一个程序中能够含有多个图,但大多数程序也只须要一个图就够了。咱们能够在多个session中重复使用一个图,可是不能在一个session中调用多个图。TensorFlow会建立默认图,你也能够自行建立新图并设为默认。显式地建立会话和图保证资源在不须要的时候合理释放。
with tf.Graph().as_default():
session_conf = tf.ConfigProto(
allow_soft_placement=FLAGS.allow_soft_placement,
log_device_placement=FLAGS.log_device_placement)
sess = tf.Session(config=session_conf)
with sess.as_default():
# Code that operates on the default graph and session comes here...
复制代码
allow_soft_placement设置容许TensorFlow在指定设备不存在时自动调整设备。例如,若是咱们的代码把一个操做放在GPU上,但又在一台没有GPU的机器上运行,若是没有allow_soft_placement
就会报错。
若是设置了log_device_placement,TensorFlow日志就会在指定设备(CPU or GPU)上存储日志文件。这对debug颇有帮助。FLAGS
是程序接收的命令行参数。
当咱们实例化TextCNN模型,全部定义的变量和操做就会被放进默认的计算图和会话。
cnn = TextCNN(
sequence_length=x_train.shape[1],
num_classes=2,
vocab_size=len(vocabulary),
embedding_size=FLAGS.embedding_dim,
filter_sizes=map(int, FLAGS.filter_sizes.split(",")),
num_filters=FLAGS.num_filters)
复制代码
接下来或作的是优化网络损失函数。TensorFlow内置几种优化器,这里使用Adam优化器。
global_step = tf.Variable(0, name="global_step", trainable=False)
optimizer = tf.train.AdamOptimizer(1e-4)
grads_and_vars = optimizer.compute_gradients(cnn.loss)
train_op = optimizer.apply_gradients(grads_and_vars, global_step=global_step)
复制代码
train_op
是新建的操做,用来对参数作梯度更新,每一次运行train_op
就是一次训练。TensorFlow会自动识别出哪些参数是“可训练的”,而后计算他们的梯度。定义了global_step
变量并传入优化器,就可让TensorFlow来完成计数。每运行一次train_op
,global_step
就+1。
(并非本文的汇总){:tensorflow:}有个概念叫summaries,让用户可以追踪并可视化训练和评估过程。好比你可能想知道损失函数和精确度随着时间的变化。更复杂的量也能够检测,好比激活层的柱状图, Summaries是系列化的对象,经过SummaryWriter写入硬盘。
# Output directory for models and summaries
timestamp = str(int(time.time()))
out_dir = os.path.abspath(os.path.join(os.path.curdir, "runs", timestamp))
print("Writing to {}\n".format(out_dir))
# Summaries for loss and accuracy
loss_summary = tf.scalar_summary("loss", cnn.loss)
acc_summary = tf.scalar_summary("accuracy", cnn.accuracy)
# Train Summaries
train_summary_op = tf.merge_summary([loss_summary, acc_summary])
train_summary_dir = os.path.join(out_dir, "summaries", "train")
train_summary_writer = tf.train.SummaryWriter(train_summary_dir, sess.graph_def)
# Dev summaries
dev_summary_op = tf.merge_summary([loss_summary, acc_summary])
dev_summary_dir = os.path.join(out_dir, "summaries", "dev")
dev_summary_writer = tf.train.SummaryWriter(dev_summary_dir, sess.graph_def)
复制代码
这里咱们分别追踪训练和评估的汇总,有些量是重复的,但又不少量是只在训练过程当中想看的(好比参数更新值)。tf.merge_summary函数能够很方便地把合并多个汇总融合到一个操做。
另外一个TensorFlow的特性是checkpointing——存储模型参数,以备不时之需。检查点能够用来继续以前中断的训练,或者提早结束获取最佳参数。检查点是经过Saver对象存储的。
# Checkpointing
checkpoint_dir = os.path.abspath(os.path.join(out_dir, "checkpoints"))
checkpoint_prefix = os.path.join(checkpoint_dir, "model")
# Tensorflow assumes this directory already exists so we need to create it
if not os.path.exists(checkpoint_dir):
os.makedirs(checkpoint_dir)
saver = tf.train.Saver(tf.all_variables())
复制代码
在训练模型以前咱们须要初始化计算图中全部的参数:
sess.run(tf.initialize_all_variables())
复制代码
initialize_all_variables很方便,一次性初始化全部变量。
如今来定义一个单独的训练步,来评估模型在一批数据上的表现,并相应地更新参数。
def train_step(x_batch, y_batch):
""" A single training step """
feed_dict = {
cnn.input_x: x_batch,
cnn.input_y: y_batch,
cnn.dropout_keep_prob: FLAGS.dropout_keep_prob
}
_, step, summaries, loss, accuracy = sess.run(
[train_op, global_step, train_summary_op, cnn.loss, cnn.accuracy],
feed_dict)
time_str = datetime.datetime.now().isoformat()
print("{}: step {}, loss {:g}, acc {:g}".format(time_str, step, loss, accuracy))
train_summary_writer.add_summary(summaries, step)
复制代码
feed_dict
里的数据将经过占位符节点送给神经网络,必须让全部节点都有值,不然TensorFlow又要报错了。另外一个输入数据的方法是queues,本文先不讨论。
接下来,咱们经过session.run()
运行train_op
,返回值就是咱们想我评估的操做结果。注意train_op
自己没有返回值,它只是更新了网络参数。最后咱们打印出本轮训练的损失函数和精确度,存储汇总至磁盘。loss和accuracy在不一样的batch之间可能差别很是大,由于咱们的batch_size很小。又由于使用了dropout,因此训练过程开始时的表现可能逊于评估过程。
咱们写了一个类似的函数来评估任意数据集的损失和精度,好比验证集或整个训练集。本质上这个函数和以前的同样,可是没有训练操做,也禁用了dropout。
def dev_step(x_batch, y_batch, writer=None):
""" Evaluates model on a dev set """
feed_dict = {
cnn.input_x: x_batch,
cnn.input_y: y_batch,
cnn.dropout_keep_prob: 1.0
}
step, summaries, loss, accuracy = sess.run(
[global_step, dev_summary_op, cnn.loss, cnn.accuracy],
feed_dict)
time_str = datetime.datetime.now().isoformat()
print("{}: step {}, loss {:g}, acc {:g}".format(time_str, step, loss, accuracy))
if writer:
writer.add_summary(summaries, step)
复制代码
最后呢,咱们要来写训练循环了。咱们一批一批的在数据上迭代,调用train_step
函数,而后评估并存储检查点:
atch_iter(
zip(x_train, y_train), FLAGS.batch_size, FLAGS.num_epochs)
# Training loop. For each batch...
for batch in batches:
x_batch, y_batch = zip(*batch)
train_step(x_batch, y_batch)
current_step = tf.train.global_step(sess, global_step)
if current_step % FLAGS.evaluate_every == 0:
print("\nEvaluation:")
dev_step(x_dev, y_dev, writer=dev_summary_writer)
print("")
if current_step % FLAGS.checkpoint_every == 0:
path = saver.save(sess, checkpoint_prefix, global_step=current_step)
print("Saved model checkpoint to {}\n".format(path))
复制代码
这里的batch_iter
是我写的一个帮助函数,用来给数据分批,tf.train.global_step
能够返回global_step
的值。训练过程的完整代码可见这里。
咱们的训练脚本把汇总写到输出路径,而后把TensorBoard指向那个路径,就能够可视化计算图和汇总。
tensorboard --logdir /PATH_TO_CODE/runs/1449760558/summaries/
复制代码
用默认参数训练(128维词嵌入,卷积核尺寸三、四、5,dropout率为0.5,每种尺寸的卷积核各128个)产出的是以下损失与精度表(蓝色为训练数据,红色是10%的测试数据)。
从图上能够看出几个事:
有几个练习能够帮助优化咱们的模型: