def masked_softmax(X, valid_lens): if valid_lens is None: return nn.functional.softmax(X,dim=-1) else: shape = X.shape if valid_lens.dim() == 1: valid_lens = torch.repeat_interleave(valid_lens, shape[1]) else: valid_lens = valid_lens.reshape(-1) X = d2l.sequence_mask(X.reshape(-1, shape[-1]), valid_lens, value=-1e6) return nn.functional.softmax(X.reshape(shape), dim=-1)
时间: 2023-11-26 14:06:21 浏览: 71
这是一个 PyTorch 中的函数 `masked_softmax`,它实现了 softmax 操作并考虑了有效长度(有效长度指的是在一个 batch 中,每个序列的真实长度,可能由于 padding 的存在而不同)。具体地,该函数的输入为一个形状为 `(batch_size, seq_len, n_classes)` 的张量 `X` 和一个形状为 `(batch_size,)` 或 `(batch_size, seq_len)` 的张量 `valid_lens`。如果 `valid_lens` 是 `None`,则表示所有序列都是有效的,直接对 `X` 进行 softmax 操作;否则,将 `valid_lens` 扩展为 `(batch_size, seq_len)` 的形状(如果原来就是这个形状则不变),并使用 `d2l.sequence_mask` 函数将超过有效长度部分的值设置为一个极大的负数,然后再对处理后的张量进行 softmax 操作。
其中,`d2l.sequence_mask` 函数实现了对一个张量中超过有效长度部分的值进行掩码的功能。具体地,对于一个形状为 `(batch_size, seq_len)` 的张量 `X` 和一个形状为 `(batch_size,)` 的张量 `valid_lens`,`d2l.sequence_mask(X, valid_lens, value=0)` 返回一个形状和 `X` 相同的张量,其中超过 `valid_lens` 的列的值都被设置为 `value`(默认为 0)。
相关问题
class MHAlayer(nn.Module): def __init__(self, n_heads, cat, input_dim, hidden_dim, attn_dropout=0.1, dropout=0): super(MHAlayer, self).__init__() self.n_heads = n_heads self.input_dim = input_dim self.hidden_dim = hidden_dim self.head_dim = self.hidden_dim / self.n_heads self.dropout = nn.Dropout(attn_dropout) self.dropout1 = nn.Dropout(dropout) self.norm = 1 / math.sqrt(self.head_dim) self.w = nn.Linear(input_dim * cat, hidden_dim, bias=False) self.k = nn.Linear(input_dim, hidden_dim, bias=False) self.v = nn.Linear(input_dim, hidden_dim, bias=False) self.fc = nn.Linear(hidden_dim, hidden_dim, bias=False) def forward(self, state_t, context, mask): ''' :param state_t: (batch_size,1,input_dim*3(GATembeding,fist_node,end_node)) :param context: (batch_size,n_nodes,input_dim) :param mask: selected nodes (batch_size,n_nodes) :return: ''' batch_size, n_nodes, input_dim = context.size() Q = self.w(state_t).view(batch_size, 1, self.n_heads, -1) K = self.k(context).view(batch_size, n_nodes, self.n_heads, -1) V = self.v(context).view(batch_size, n_nodes, self.n_heads, -1) Q, K, V = Q.transpose(1, 2), K.transpose(1, 2), V.transpose(1, 2) compatibility = self.norm * torch.matmul(Q, K.transpose(2, 3)) compatibility = compatibility.squeeze(2) mask = mask.unsqueeze(1).expand_as(compatibility) u_i = compatibility.masked_fill(mask.bool(), float("-inf")) scores = F.softmax(u_i, dim=-1) scores = scores.unsqueeze(2) out_put = torch.matmul(scores, V) out_put = out_put.squeeze(2).view(batch_size, self.hidden_dim) out_put = self.fc(out_put) return out_put
这是一个PyTorch的神经网络模型定义,其中定义了一个叫做MHAlayer的模块。这个模块实现了Multi-Head Attention机制,用于文本序列处理中的Attention机制。具体来说,这个模块输入了三个张量:state_t,context和mask,其中state_t是一个(batch_size,1,input_dim*3)的张量,context是一个(batch_size,n_nodes,input_dim)的张量,mask是一个(batch_size,n_nodes)的张量,表示需要进行Attention的节点。在模块中,首先用线性层将state_t和context分别映射到hidden_dim维的向量,并将输入的维度变为(batch_size, n_heads, hidden_dim),然后计算查询向量Q,键向量K和值向量V,并将它们都分成n_heads份。接下来,计算对应的Attention得分compatibility,并根据mask将需要Attention的节点的得分设置为负无穷。然后对得分进行softmax归一化,并加权求和得到Attention的输出。最后再通过一个线性层转换维度,并返回输出。
import torchimport torch.nn as nnclass MultiHeadAttention(nn.Module): def __init__(self, d_model, num_heads): super(MultiHeadAttention, self).__init__() self.num_heads = num_heads self.d_model = d_model assert d_model % self.num_heads == 0 self.depth = d_model // self.num_heads self.Wq = nn.Linear(d_model, d_model) self.Wk = nn.Linear(d_model, d_model) self.Wv = nn.Linear(d_model, d_model) self.fc = nn.Linear(d_model, d_model) def scaled_dot_product_attention(self, Q, K, V, mask=None): d_k = Q.size(-1) scores = torch.matmul(Q, K.transpose(-1, -2)) / torch.sqrt(torch.tensor(d_k, dtype=torch.float32)) if mask is not None: scores = scores.masked_fill(mask == 0, -1e9) attention = torch.softmax(scores, dim=-1) output = torch.matmul(attention, V) return output, attention def split_heads(self, x, batch_size): x = x.view(batch_size, -1, self.num_heads, self.depth) return x.permute(0, 2, 1, 3) def forward(self, Q, K, V, mask=None): batch_size = Q.size(0) Q = self.Wq(Q) K = self.Wk(K) V = self.Wv(V) Q = self.split_heads(Q, batch_size) K = self.split_heads(K, batch_size) V = self.split_heads(V, batch_size) scaled_attention, attention = self.scaled_dot_product_attention(Q, K, V, mask) scaled_attention = scaled_attention.permute(0, 2, 1, 3).contiguous() scaled_attention = scaled_attention.view(batch_size, -1, self.d_model) output = self.fc(scaled_attention) return output, attention
上述代码是一个用PyTorch实现的多头注意力机制(Multi-Head Attention)的模块,该模块可以被用来构建神经网络模型。它的参数有:
- d_model:表示输入向量的维度,也就是embedding的维度。
- num_heads:表示attention头的数量。
输入的维度是:
- Q, K, V:三个输入张量的维度都为 [batch_size, seq_length, d_model],其中batch_size代表batch的大小,seq_length代表输入序列的长度,d_model代表输入向量的维度。
- mask:一个shape为[batch_size, 1, seq_length, seq_length]的张量,用于掩盖无效的位置,使得在计算注意力矩阵时,注意力矩阵中无效位置对应的权重为0。如果没有无效位置,则可以设置为None。
阅读全文