87.序列到序列学习(seq2seq)以及代码实现

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6

1. 机器翻译

在这里插入图片描述

2. Seq2Seq

在这里插入图片描述

双向RNN可以做encoder但不能做decoder。

3. 编码器-解码器细节

在这里插入图片描述

4. 训练

在这里插入图片描述

5. 衡量生成序列的好坏的BLEU

在这里插入图片描述

上面的公式既加入了段序列的惩罚项又加入了更难出现的长序列的高权重。

6. 总结

  • Seq2seq从一个句子生成另一个句子
  • 编码器和解码器都是RNN
  • 将编码器最后时间隐状态来初始解码器隐状态来完成信息传递
  • 常用BLEU来衡量生成序列的好坏

7. 代码实现

下面我们动手构建 seq2seq的设计 并将基于“英-法”数据集来训练这个机器翻译模型。

import collections
import math
import torch
from torch import nn
from d2l import torch as d2l

7.1 编码器

现在来实现循环神经网络编码器。 注意我们使用了嵌入层embedding layer 来获得输入序列中每个词元的特征向量。 嵌入层的权重是一个矩阵 其行数等于输入词表的大小vocab_size 其列数等于特征向量的维度embed_size。 对于任意输入词元的索引 𝑖 嵌入层获取权重矩阵的第 𝑖 行从 0 开始以返回其特征向量。 另外本文选择了一个多层门控循环单元来实现编码器。

class Seq2SeqEncoder(d2l.Encoder):
    """用于序列到序列学习的循环神经网络编码器"""
    def __init__(self, vocab_size, embed_size, num_hiddens, num_layers,
                 dropout=0, **kwargs):
        super(Seq2SeqEncoder, self).__init__(**kwargs)
        # 嵌入层
        # embed是word2vec的思想把字典里one-hot编码的字或者词变成预训练可能需要微调的词向量
        # embed就是将word映射到向量空间
        self.embedding = nn.Embedding(vocab_size, embed_size)
        # 第一个参数是inputs大小在之前一些代码中传入的是vocab_size或者lenvocab是词表的大小
        # 但是在这里因为使用了nn.Embedding将词表的大小改变成了embed_size再作为输入
        self.rnn = nn.GRU(embed_size, num_hiddens, num_layers,
                          dropout=dropout)

    def forward(self, X, *args):
        # 输出'X'的形状(batch_size,num_steps,embed_size)
        X = self.embedding(X)
        # 在循环神经网络模型中第一个轴对应于时间步
        # 把batch_size换到中间把num_steps换到第一个轴
        # 转换为时间步数量批量大小词表大小
        X = X.permute(1, 0, 2)
        # 如果未提及状态则默认为0

        # 输出output是每一个时间步的最后一层RNN的输出根据图去理解最后一层RNN往上的输出
        # state 在最后一个时刻所有的层的输出根据图去理解是每一层最右边的输出
        output, state = self.rnn(X)
        # output的形状:(num_steps,batch_size,num_hiddens)
        # state的形状:(num_layers,batch_size,num_hiddens)
        return output, state

下面我们实例化上述编码器的实现 我们使用一个两层门控循环单元编码器其隐藏单元数为 16 。 给定一小批量的输入序列X批量大小为 4 时间步为 7 。 在完成所有时间步后 最后一层的隐状态的输出是一个张量output由编码器的循环层返回 其形状为时间步数批量大小隐藏单元数。

Pytorchmodel.train()和model.eval()用法和区别以及model.eval()和torch.no_grad()的区别

encoder = Seq2SeqEncoder(vocab_size=10, embed_size=8, num_hiddens=16,
                         num_layers=2)
encoder.eval() # 在eval模式下dropout不会生效
X = torch.zeros((4, 7), dtype=torch.long)
output, state = encoder(X)
output.shape

运行结果

在这里插入图片描述

由于这里使用的是门控循环单元 所以在最后一个时间步的多层隐状态的形状是 隐藏层的数量批量大小隐藏单元的数量。 如果使用长短期记忆网络state中还将包含记忆单元信息。

state.shape

运行结果

在这里插入图片描述

7.2 解码器

当实现解码器时 我们直接使用编码器最后一个时间步的隐状态来初始化解码器的隐状态。 这就要求使用循环神经网络实现的编码器和解码器具有相同数量的层和隐藏单元。 为了进一步包含经过编码的输入序列的信息上下文变量在所有的时间步与解码器的输入进行拼接(concatenate)。 为了预测输出词元的概率分布 在循环神经网络解码器的最后一层使用全连接层来变换隐状态。

class Seq2SeqDecoder(d2l.Decoder):
    """用于序列到序列学习的循环神经网络解码器"""
    def __init__(self, vocab_size, embed_size, num_hiddens, num_layers,
                 dropout=0, **kwargs):
        super(Seq2SeqDecoder, self).__init__(**kwargs)
        # decoder在模型上和encoder是一样的
        # decoder有自己的embedding层不能和encoder共享因为词汇都不一样
        self.embedding = nn.Embedding(vocab_size, embed_size)
        # 这里的输入是embed_size + num_hiddens
        # 并且假设了encoder隐藏层大小和decoder的隐藏层大小一样
        self.rnn = nn.GRU(embed_size + num_hiddens, num_hiddens, num_layers,
                          dropout=dropout)
        # decoder有输出层
        self.dense = nn.Linear(num_hiddens, vocab_size)

    def init_state(self, enc_outputs, *args):
      # enc_outputs是output、state
      # enc_outputs[1] 就是encoder输出的state
        return enc_outputs[1]

    def forward(self, X, state):
        # 输出'X'的形状(batch_size,num_steps,embed_size)
        X = self.embedding(X).permute(1, 0, 2)

        # context是上下文信息
        # state是最后一个时刻的所有RNN层的隐藏状态也就是图中最后一竖的H
        # state[-1]就是最后一个时刻的最后一层RNN的输出也就是图中右上角的H
        # 那个右上角的H包括了所有浓缩的信息把它拿到之后重复几次
        # 重复成decoder输入的长度每个时刻都重复一次
        # 广播context使其具有与X相同的num_steps
        # 从另一个角度理解为repeat操作增加通道数这里把二位矩阵扩充到三维增加了seq维
        context = state[-1].repeat(X.shape[0], 1, 1)
        # decoder中RNN的输入是当前embedding的输出加上encoder传过来的上下文信息
        # 虽然state已经传过来了但是觉得不够还要把最后那个时刻context和embedding拼在一起作为输入
        # 这也是为什么decoder的RNN的输入是embed_size + num_hiddens
        X_and_context = torch.cat((X, context), 2)
        output, state = self.rnn(X_and_context, state)
        output = self.dense(output).permute(1, 0, 2)
        # output的形状:(batch_size,num_steps,vocab_size)
        # state的形状:(num_layers,batch_size,num_hiddens)
        return output, state

下面我们用与前面提到的编码器中相同的超参数来实例化解码器。 如我们所见解码器的输出形状变为批量大小时间步数词表大小 其中张量的最后一个维度存储预测的词元分布。

decoder = Seq2SeqDecoder(vocab_size=10, embed_size=8, num_hiddens=16,
                         num_layers=2)
decoder.eval()
state = decoder.init_state(encoder(X))
output, state = decoder(X, state)
# 输出形状为批量大小时间步数词表大小
# 对每一个样本的每一个时刻都做一个输出
# state的形状为层数批量大小隐藏层大小
output.shape, state.shape

运行结果

在这里插入图片描述

7.3 损失函数

在每个时间步解码器预测了输出词元的概率分布。 类似于语言模型可以使用softmax来获得分布 并通过计算交叉熵损失函数来进行优化。 回想一下machine_translation中 特定的填充词元被添加到序列的末尾 因此不同长度的序列可以以相同形状的小批量加载。 但是我们应该将填充词元的预测排除在损失函数的计算之外

为此我们可以使用下面的sequence_mask函数 通过零值化屏蔽不相关的项 以便后面任何不相关预测的计算都是与零的乘积结果都等于零。 例如如果两个序列的有效长度不包括填充词元分别为 1 和 2 则第一个序列的第一项和第二个序列的前两项之后的剩余项将被清除为零。

def sequence_mask(X, valid_len, value=0):
    """在序列中屏蔽不相关的项"""
    maxlen = X.size(1)
    mask = torch.arange((maxlen), dtype=torch.float32,
                        device=X.device)[None, :] < valid_len[:, None]
    X[~mask] = value
    return X

X = torch.tensor([[1, 2, 3], [4, 5, 6]])
sequence_mask(X, torch.tensor([1, 2]))

我们还可以使用此函数屏蔽最后几个轴上的所有项。如果愿意也可以使用指定的非零值来替换这些项。

X = torch.ones(2, 3, 4)
sequence_mask(X, torch.tensor([1, 2]), value=-1)

现在我们可以通过扩展softmax交叉熵损失函数遮蔽不相关的预测。 最初所有预测词元的掩码都设置为1。 一旦给定了有效长度与填充词元对应的掩码将被设置为0。 最后将所有词元的损失乘以掩码以过滤掉损失中填充词元产生的不相关预测。

class MaskedSoftmaxCELoss(nn.CrossEntropyLoss):
    """带遮蔽的softmax交叉熵损失函数"""
    # pred的形状(batch_size,num_steps,vocab_size)
    # label的形状(batch_size,num_steps)
    # valid_len的形状(batch_size,)
    def forward(self, pred, label, valid_len):
        # 生成和label形状一样全1的矩阵
        weights = torch.ones_like(label)
        # 把有效的长度保留下来其他变为0
        weights = sequence_mask(weights, valid_len)
        # reduction定义为none就不会对loss求和或者求平均
        self.reduction='none'
        # 这里loss调用的是父类函数其实super中的内容可以删掉
        # 在pytorch中的MaskedSoftmaxCELoss规定要把vocab_size放在第2个维度
        unweighted_loss = super(MaskedSoftmaxCELoss, self).forward(
            pred.permute(0, 2, 1), label)
        # unweighted_loss * weights 会使得有效的地方留下来其他为0
        # dim=1就是对每个句子取平均
        weighted_loss = (unweighted_loss * weights).mean(dim=1)
        return weighted_loss

我们可以创建三个相同的序列来进行代码健全性检查 然后分别指定这些序列的有效长度为 4 、 2 和 0 。 结果就是第一个序列的损失应为第二个序列的两倍而第三个序列的损失应为零。

loss = MaskedSoftmaxCELoss()
# 3是批量大小4是时间步数10是每个单词向量的维度
# torch.tensor([4, 2, 0]第一个样本所有都是valid第二个样本只有前两个是valie
# 最后一个样本全都不是valid
loss(torch.ones(3, 4, 10), torch.ones((3, 4), dtype=torch.long),
     torch.tensor([4, 2, 0]))

运行结果

在这里插入图片描述

7.4 训练

在下面的循环训练过程中 特定的序列开始词元“< bos>”和 原始的输出序列不包括序列结束词元“< eos>” 拼接在一起作为解码器的输入。 这被称为强制教学teacher forcing 因为原始的输出序列词元的标签被送入解码器。 或者将来自上一个时间步的预测得到的词元作为解码器的当前输入。

def train_seq2seq(net, data_iter, lr, num_epochs, tgt_vocab, device):
    """训练序列到序列模型"""
    def xavier_init_weights(m):
        if type(m) == nn.Linear:
            nn.init.xavier_uniform_(m.weight)
        if type(m) == nn.GRU:
            for param in m._flat_weights_names:
                if "weight" in param:
                    nn.init.xavier_uniform_(m._parameters[param])

    # net就是encoder-decoder
    net.apply(xavier_init_weights)
    net.to(device)
    optimizer = torch.optim.Adam(net.parameters(), lr=lr)
    loss = MaskedSoftmaxCELoss()
    net.train()
    animator = d2l.Animator(xlabel='epoch', ylabel='loss',
                     xlim=[10, num_epochs])
    for epoch in range(num_epochs):
        timer = d2l.Timer()
        metric = d2l.Accumulator(2)  # 训练损失总和词元数量
        for batch in data_iter:
          # batch中有源句子源句子的valid_len目标句子目标句子的valid_len
            optimizer.zero_grad()
            X, X_valid_len, Y, Y_valid_len = [x.to(device) for x in batch]
            # bosbegin of sentence源句子要翻译需要这个标志
            bos = torch.tensor([tgt_vocab['<bos>']] * Y.shape[0],
                          device=device).reshape(-1, 1)
            # dec_input 就是吧bos和Ytarget里面的除最后一项组合在一起
            dec_input = torch.cat([bos, Y[:, :-1]], 1)  # 强制教学
            # net的输入是编码器输入、解码器输入、编码器有效长度
            Y_hat, _ = net(X, dec_input, X_valid_len)
            # 计算loss的时候因为y的填充不要算loss所以传入Y_valid_len告诉非填充部分
            l = loss(Y_hat, Y, Y_valid_len)
            l.sum().backward()	# 损失函数的标量进行“反向传播”
            d2l.grad_clipping(net, 1)
            num_tokens = Y_valid_len.sum()
            optimizer.step()
            with torch.no_grad():
                metric.add(l.sum(), num_tokens)
        if (epoch + 1) % 10 == 0:
            animator.add(epoch + 1, (metric[0] / metric[1],))
    print(f'loss {metric[0] / metric[1]:.3f}, {metric[1] / timer.stop():.1f} '
        f'tokens/sec on {str(device)}')

现在在机器翻译数据集上我们可以 创建和训练一个循环神经网络“编码器-解码器”模型用于序列到序列的学习。

embed_size, num_hiddens, num_layers, dropout = 32, 32, 2, 0.1
batch_size, num_steps = 64, 10 # 句子长度为10
lr, num_epochs, device = 0.005, 300, d2l.try_gpu()

train_iter, src_vocab, tgt_vocab = d2l.load_data_nmt(batch_size, num_steps)
encoder = Seq2SeqEncoder(len(src_vocab), embed_size, num_hiddens, num_layers,
                        dropout)
decoder = Seq2SeqDecoder(len(tgt_vocab), embed_size, num_hiddens, num_layers,
                        dropout)
# 用encoder和decoder做出net
net = d2l.EncoderDecoder(encoder, decoder)
train_seq2seq(net, train_iter, lr, num_epochs, tgt_vocab, device)

运行结果

训练的速度很快是因为encoder和decoder都是一个RNN长度为10总共就2个RNN也能看做是一个长度为20的RNN。

在这里插入图片描述

7.5 预测

为了采用一个接着一个词元的方式预测输出序列 每个解码器当前时间步的输入都将来自于前一时间步的预测词元。 与训练类似序列开始词元“< bos>” 在初始时间步被输入到解码器中。 该预测过程如 图所示 当输出序列的预测遇到序列结束词元“< eos>”时预测就结束了。

在这里插入图片描述

使用循环神经网络编码器-解码器逐词元地预测输出序列。

# 之前训练的时候在做解码器的输入和输出时输入用的是<bos>+真实的target句子
# 而预测的时候我们是不知道真实句子的所以会有区别
def predict_seq2seq(net, src_sentence, src_vocab, tgt_vocab, num_steps,
                    device, save_attention_weights=False):
    """序列到序列模型的预测"""
    # 在预测时将net设置为评估模式
    net.eval()
    # 把 源句子+<eos> 转换为idx
    src_tokens = src_vocab[src_sentence.lower().split(' ')] + [
        src_vocab['<eos>']]
    # 有效长度
    enc_valid_len = torch.tensor([len(src_tokens)], device=device)
    # 对源句子进行填充和截取
    src_tokens = d2l.truncate_pad(src_tokens, num_steps, src_vocab['<pad>'])
    # 添加批量轴
    enc_X = torch.unsqueeze(
        torch.tensor(src_tokens, dtype=torch.long, device=device), dim=0)
    enc_outputs = net.encoder(enc_X, enc_valid_len)
    # 至此以上代码都是encoder部分
    dec_state = net.decoder.init_state(enc_outputs, enc_valid_len)
    # 添加批量轴
    # 这里和之前有区别之前给的是<bos>+目标句子而在这里就是给<bos>
    dec_X = torch.unsqueeze(torch.tensor(
        [tgt_vocab['<bos>']], dtype=torch.long, device=device), dim=0)
    output_seq, attention_weight_seq = [], []
    for _ in range(num_steps): # 预测n步
        # 每一步都往decoder中放入dec_X和dec_state
        # 第一次循环的dec_X就是<bos>
        # 输出Y和更新的state
        Y, dec_state = net.decoder(dec_X, dec_state)
        # 我们使用具有预测最高可能性的词元作为解码器在下一时间步的输入
        # dim=2表示的是vocab维此时的dec_X是作为下一步的输入
        dec_X = Y.argmax(dim=2)
        # 因为dec_X的维度比较高通过squeeze
        pred = dec_X.squeeze(dim=0).type(torch.int32).item()
        # 保存注意力权重稍后讨论
        if save_attention_weights:
            attention_weight_seq.append(net.decoder.attention_weights)
        # 一旦序列结束词元被预测输出序列的生成就完成了
        if pred == tgt_vocab['<eos>']:
            break
        # 没有预测结束就把预测结果放入output_seq
        output_seq.append(pred)
    # 把预测的output_seq通过to_tokens把token词元查出来并且用空格拼接就能变成一句话了
    return ' '.join(tgt_vocab.to_tokens(output_seq)), attention_weight_seq

7.6 预测序列的评估

在这里插入图片描述

BLEU的代码实现如下

def bleu(pred_seq, label_seq, k):
    """计算BLEU"""
    pred_tokens, label_tokens = pred_seq.split(' '), label_seq.split(' ')
    len_pred, len_label = len(pred_tokens), len(label_tokens)
    # 公式的第一部分
    score = math.exp(min(0, 1 - len_label / len_pred))
    for n in range(1, k + 1): # n-gram从1一直算到k元语法
        num_matches, label_subs = 0, collections.defaultdict(int)
        for i in range(len_label - n + 1):
            label_subs[' '.join(label_tokens[i: i + n])] += 1
        for i in range(len_pred - n + 1):
            if label_subs[' '.join(pred_tokens[i: i + n])] > 0:
                num_matches += 1
                label_subs[' '.join(pred_tokens[i: i + n])] -= 1
        score *= math.pow(num_matches / (len_pred - n + 1), math.pow(0.5, n))
    return score

最后利用训练好的循环神经网络“编码器-解码器”模型 将几个英语句子翻译成法语并计算BLEU的最终结果。

engs = ['go .', "i lost .", 'he\'s calm .', 'i\'m home .']
fras = ['va !', 'j\'ai perdu .', 'il est calme .', 'je suis chez moi .']
for eng, fra in zip(engs, fras):
    translation, attention_weight_seq = predict_seq2seq(
        net, eng, src_vocab, tgt_vocab, num_steps, device)
    print(f'{eng} => {translation}, bleu {bleu(translation, fra, k=2):.3f}')

运行结果

在这里插入图片描述

这节课整个过程可以理解为个人理解有问题欢迎指正【摘自b站评论】

1.获得训练集样本将样本转化为embeding矩阵每一个embeding向量对应一个词embeding向量的种类有vocab_size个,这些embeding向量重复累积构成了整个文本(二维张量)。代码中的输入X是(batch_size,num_steps,embed_size)意义即每个词用一个embeding向量表示这个向量的维度为embed_size每个时间步有num_steps个词取batch_size个批量。

2.确定超参数时间步s时间步长度即为单次输入(英语)和输出(法语)的最大数量。

3.确定批量b这里是为了优化以便训练时更快更好地迭代收敛。

4.每次选取b组长度为s的样本这些样本在encoder中前向传递最终得到一个H把H和对应的法语张量合并共同作为decoder的输入后续像rnn一样前向传递即将当前词和此时的状态共同作为输入来预测下一个词。

5.预测完成后将output和实际文本对应的张量进行交叉熵计算计算时只取有效长度避免出现多余的损失值。

6.训练完成后即可输入英语把输出结果累积得到英语对应的法语翻译并用bleu衡量翻译的好坏。

8. Q&A

Q1:encoder输出和decoder的输入拼接和按位相加起来有什么区别吗

A1: 不能按位加因为decoder的输入是 embedding size而encoder的输出是hidden size上面的代码是取的一样的值但实际上不能这么做因为长度不一样。

Q2: 实际句子的长度超过了设定的句子长度是直接截掉不用还是放到下一个句子

A2: 截掉不用

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6