当前位置: 首页 > news >正文

Encoder-Decoder:Seq2seq

目录

  • 一、编码器解码器架构:
    • 1.定义:
    • 2.在CNN中的体现:
    • 3.在RNN中的体现:
    • 4.代码:
  • 二、Seq2seq:
    • 1.模型架构:
      • 1.1编码器:
      • 1.2解码器:
    • 2.架构细节:
    • 3.模型评估指标BLEU:
    • 4.代码:
  • 三、束搜索:
    • 1.贪心搜索:
    • 2.束搜索:

一、编码器解码器架构:

1.定义:

在这里插入图片描述

Encoder负责对Input进行特征提取,输出特征矩阵State
Decoder接收State,负责进行预测并输出

2.在CNN中的体现:

在这里插入图片描述

3.在RNN中的体现:

在这里插入图片描述

4.代码:

from torch import nnclass Encoder(nn.Module):"""编码器-解码器结构的基本编码器接口。"""def __init__(self, **kwargs):super(Encoder, self).__init__(**kwargs)def forward(self, X, *args):raise NotImplementedErrorclass Decoder(nn.Module):"""编码器-解码器结构的基本解码器接口。"""def __init__(self, **kwargs):super(Decoder, self).__init__(**kwargs)def init_state(self, enc_outputs, *args):raise NotImplementedErrordef forward(self, X, state):raise NotImplementedErrorclass EncoderDecoder(nn.Module):"""编码器-解码器结构的基类。"""def __init__(self, encoder, decoder, **kwargs):super(EncoderDecoder, self).__init__(**kwargs)self.encoder = encoderself.decoder = decoderdef forward(self, enc_X, dec_X, *args):enc_outputs = self.encoder(enc_X, *args)dec_state = self.decoder.init_state(enc_outputs, *args)return self.decoder(dec_X, dec_state)    

二、Seq2seq:

1.模型架构:

这里以机器翻译任务为例:
在这里插入图片描述

1.1编码器:

编码器不管在训练阶段还是预测阶段都是用于提取特征,可以是单层RNN、多层RNN、双向RNN(双向RNN不仅可以提取上文序列特征,还可以提取下文的序列特征)

1.2解码器:

解码器在不同阶段作用不同,只能是单层RNN或多层RNN,不能是双向RNN(解码器用于预测,双向RNN不能预测)

  • 训练阶段,解码器主要是为了特征提取,通过接收编码器的输出隐藏状态作为h0并接收预测的真实值Input,每个时间步使用隐藏状态ht-1进行特征提取并更新隐藏状态ht,然后将ht和当前时间步的真实值token(而非预测值,因为是要更好的学习)作为下一个时间步的输入,不断更新可学习参数。
  • 预测阶段,解码器主要是为了执行预测任务,不再接收预测的真实值(因为不知道),仅接收编码器的输出隐藏状态作为h0,每个时间步使用隐藏状态ht-1进行预测并更新隐藏状态ht,然后将ht和当前时间步的预测值token作为下一个时间步的输入,进行下一个token的预测。

2.架构细节:

Seq2seq的编码器和解码器都是RNN
在这里插入图片描述

3.模型评估指标BLEU:

在这里插入图片描述

4.代码:

import collections
import math
import torch
from torch import nn
from d2l import torch as d2l# 使用GRU作为编码器
class Seq2SeqEncoder(d2l.Encoder):def __init__(self, vocab_size, embed_size, num_hiddens, num_layers,dropout=0, **kwargs):super(Seq2SeqEncoder, self).__init__(**kwargs)# 1self.embedding = nn.Embedding(vocab_size, embed_size)# 2 self.rnn = nn.GRU(embed_size, num_hiddens, num_layers,dropout=dropout)def forward(self, X, *args):X = self.embedding(X)X = X.permute(1, 0, 2)output, state = self.rnn(X)return output, state# 使用GRU作为解码器
class Seq2SeqDecoder(d2l.Decoder):def __init__(self, vocab_size, embed_size, num_hiddens, num_layers,dropout=0, **kwargs):super(Seq2SeqDecoder, self).__init__(**kwargs)# 1self.embedding = nn.Embedding(vocab_size, embed_size)# 2self.rnn = nn.GRU(embed_size + num_hiddens, num_hiddens, num_layers,dropout=dropout)# 3self.dense = nn.Linear(num_hiddens, vocab_size)def init_state(self, enc_outputs, *args):return enc_outputs[1]def forward(self, X, state):X = self.embedding(X).permute(1, 0, 2)context = state[-1].repeat(X.shape[0], 1, 1)X_and_context = torch.cat((X, context), 2)output, state = self.rnn(X_and_context, state)output = self.dense(output).permute(1, 0, 2)return output, state# 在序列中屏蔽不相关的项,即屏蔽序列中之前使用<pad>填充的无效值   
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] = valuereturn X# 填充的无效值不参与损失值的计算,因为这些值的预测对错没有意义
class MaskedSoftmaxCELoss(nn.CrossEntropyLoss):def forward(self, pred, label, valid_len):weights = torch.ones_like(label)weights = sequence_mask(weights, valid_len)self.reduction = 'none'unweighted_loss = super(MaskedSoftmaxCELoss,self).forward(pred.permute(0, 2, 1), label)weighted_loss = (unweighted_loss * weights).mean(dim=1)return weighted_loss# 训练过程
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.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:X, X_valid_len, Y, Y_valid_len = [x.to(device) for x in batch]bos = torch.tensor([tgt_vocab['<bos>']] * Y.shape[0],device=device).reshape(-1, 1)dec_input = torch.cat([bos, Y[:, :-1]], 1)Y_hat, _ = net(X, dec_input, X_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)}')# 预测过程    
def predict_seq2seq(net, src_sentence, src_vocab, tgt_vocab, num_steps,device, save_attention_weights=False):net.eval()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)dec_state = net.decoder.init_state(enc_outputs, enc_valid_len)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):Y, dec_state = net.decoder(dec_X, dec_state)dec_X = Y.argmax(dim=2)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>']:breakoutput_seq.append(pred)return ' '.join(tgt_vocab.to_tokens(output_seq)), attention_weight_seq    # 模型评估指标
def bleu(pred_seq, label_seq, k):  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):num_matches, label_subs = 0, collections.defaultdict(int)for i in range(len_label - n + 1):label_subs[''.join(label_tokens[i:i + n])] += 1for i in range(len_pred - n + 1):if label_subs[''.join(pred_tokens[i:i + n])] > 0:num_matches += 1label_subs[''.join(pred_tokens[i:i + n])] -= 1score *= math.pow(num_matches / (len_pred - n + 1), math.pow(0.5, n))return score# 训练    
embed_size, num_hiddens, num_layers, dropout = 32, 32, 2, 0.1
batch_size, num_steps = 64, 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)
net = d2l.EncoderDecoder(encoder, decoder)
train_seq2seq(net, train_iter, lr, num_epochs, tgt_vocab, device)    # 预测    
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}')

三、束搜索:

1.贪心搜索:

在这里插入图片描述

2.束搜索:

在这里插入图片描述

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • Day12--Servlet实现前后端交互(案例:学生信息管理系统登录页面)
  • ZooKeeper日志自动清理实用脚本
  • AI可解释性(Python语言版)书籍推荐
  • 什么样的双筒式防爆器把煤矿吸引?
  • kalman的python实现
  • Elasticsearch模糊查询之Wildcard
  • Unity横板动作游戏 - 素材导入和整理
  • 月薪竟然高达60k,AI大模型凭什么?
  • 手摸手教你前端和后端是如何实现导出 Excel 的?
  • Python 爬虫项目实战(一):破解网易云 VIP 免费下载付费歌曲
  • uniapp h5支付(支付宝和微信支付)
  • [ Socket学习 ] 第一章:网络基础知识
  • 常用排序算法的实现与介绍
  • Pyinstaller打包OSError: could not get source code【终极解决】
  • [Meachines] [Easy] Admirer Adminer远程Mysql反向+Python三方库函数劫持权限提升
  • Laravel 中的一个后期静态绑定
  • LintCode 31. partitionArray 数组划分
  • mac修复ab及siege安装
  • MaxCompute访问TableStore(OTS) 数据
  • Python_网络编程
  • React+TypeScript入门
  • storm drpc实例
  • 短视频宝贝=慢?阿里巴巴工程师这样秒开短视频
  • 反思总结然后整装待发
  • 分布式任务队列Celery
  • 记录一下第一次使用npm
  • 适配mpvue平台的的微信小程序日历组件mpvue-calendar
  • 学习使用ExpressJS 4.0中的新Router
  • 走向全栈之MongoDB的使用
  • const的用法,特别是用在函数前面与后面的区别
  • 如何通过报表单元格右键控制报表跳转到不同链接地址 ...
  • 新海诚画集[秒速5センチメートル:樱花抄·春]
  • ​直流电和交流电有什么区别为什么这个时候又要变成直流电呢?交流转换到直流(整流器)直流变交流(逆变器)​
  • # wps必须要登录激活才能使用吗?
  • ## 基础知识
  • #nginx配置案例
  • #微信小程序(布局、渲染层基础知识)
  • ( 10 )MySQL中的外键
  • (007)XHTML文档之标题——h1~h6
  • (1)(1.11) SiK Radio v2(一)
  • (12)Linux 常见的三种进程状态
  • (13)DroneCAN 适配器节点(一)
  • (8)Linux使用C语言读取proc/stat等cpu使用数据
  • (笔记)M1使用hombrew安装qemu
  • (非本人原创)史记·柴静列传(r4笔记第65天)
  • (分享)自己整理的一些简单awk实用语句
  • (附源码)spring boot火车票售卖系统 毕业设计 211004
  • (规划)24届春招和25届暑假实习路线准备规划
  • (论文阅读笔记)Network planning with deep reinforcement learning
  • (转)socket Aio demo
  • (转载)跟我一起学习VIM - The Life Changing Editor
  • .net core 客户端缓存、服务器端响应缓存、服务器内存缓存
  • .NET+WPF 桌面快速启动工具 GeekDesk
  • .Net7 环境安装配置
  • .NET8.0 AOT 经验分享 FreeSql/FreeRedis/FreeScheduler 均已通过测试