现代循环神经网络 - 序列到序列学习

序列到序列学习(seq2seq)

在本节中,我们将使用两个循环神经网络的编码器和解码器,并将其应用于序列到序列(sequence to sequence,seq2seq)类的学习任务

遵循编码器-解码器的设计原则,循环神经网络编码器使用长度可变的序列作为输入,将其转换为固定形状的隐状态。换言之,输入序列的信息被编码到循环神经网络编码器的隐状态中。为了连续生成输出序列的词元,独立的循环神经网络解码器是基于输入序列的编码信息和输出序列已经看见的或者生成的词元来预测下一个词元。下图演示了如何在机器翻译中使用两个循环神经网络进行序列到序列的学习

在 图9.7.1中,特定的“<eos>”表⽰序列结束词元。⼀旦输出序列⽣成此词元,模型就会停⽌预测。在循环神经⽹络解码器的初始化时间步,有两个特定的设计决定:⾸先,特定的“<bos>”表⽰序列开始词元,它是解码器的输⼊序列的第⼀个词元。其次,使⽤循环神经⽹络编码器最终的隐状态来初始化解码器的隐状态。例如,在 [Sutskever et al., 2014]的设计中,正是基于这种设计将输⼊序列的编码信息送⼊到解码器中来⽣成输出序列的。在其他⼀些设计中 [Cho et al., 2014b],如 图9.7.1所⽰,编码器最终的隐状态在每⼀个时间步都作为解码器的输⼊序列的⼀部分。类似于 8.3节中语⾔模型的训练,可以允许标签成为原始的输出序列,从源序列词元“<bos>”、“Ils”、“regardent”、“.”到新序列词元“Ils”、“regardent”、“.”、“<eos>”来移动预测的位置

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

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

1 - 编码器

class Seq2SeqEncoder(d2l.Encoder):
    """用于序列到序列学习的循环神经网络编码器"""
    def __init__(self,vocab_size,embed_size,num_hiddens,num_layers,dropout=0,**kwargs):
        super(Seq2SeqEncoder,self).__init__(**kwargs)
        # 嵌入层
        self.embedding = nn.Embedding(vocab_size,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)
        # 在循环神经网络模型中,第一个轴对应于时间步
        X = X.permute(1,0,2)
        # 如果未提交状态,则默认为0
        output,state = self.rnn(X)
        # output的形状:(num_steps,batch_size,num_hiddens)
        # state[0]的形状:(num_layers,batch_size,num_hiddens)
        return output, state

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

encoder = Seq2SeqEncoder(vocab_size=10,embed_size=8,num_hiddens=16,num_layers=2)

encoder.eval()
X = torch.zeros((4,7),dtype=torch.long)
output,state = encoder(X)
output.shape
torch.Size([7, 4, 16])

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

state.shape
torch.Size([2, 4, 16])

2 - 解码器

class Seq2SeqDecoder(d2l.Decoder):
    """用于序列到序列学习的循环神经网络解码器"""
    def __init__(self,vocab_size,embed_size,num_hiddens,num_layers,dropout=0,**kwargs):
        super(Seq2SeqDecoder,self).__init__(**kwargs)
        self.embedding = nn.Embedding(vocab_size, embed_size)
        self.rnn = nn.GRU(embed_size + num_hiddens, num_hiddens, num_layers,dropout=dropout)
        self.dense = nn.Linear(num_hiddens, vocab_size)
        
    def init_state(self,enc_outputs,*args):
        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,使其具有与X相同的num_steps
        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)
        # output的形状:(batch_size,num_steps,vocab_size)
        # state[0]的形状:(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)
output.shape,state.shape
(torch.Size([4, 7, 10]), torch.Size([2, 4, 16]))

3 - 损失函数

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

为此,我们可以使用下面的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]))
tensor([[1, 0, 0],
        [4, 5, 0]])

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

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

        [[ 1.,  1.,  1.,  1.],
         [ 1.,  1.,  1.,  1.],
         [-1., -1., -1., -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):
        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

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

loss = MaskedSoftmaxCELoss()
loss(torch.ones(3, 4, 10), torch.ones((3, 4), dtype=torch.long),torch.tensor([4, 2, 0]))
tensor([2.3026, 1.1513, 0.0000])

4 - 训练

在下面的循环训练过程中,如图9.7.1所示,特定的序列开始词元<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.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:
            optimizer.zero_grad()
            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)}')

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

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)
loss 0.021, 23807.7 tokens/sec on cuda:0

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mg3wmo2n-1662904032828)(https://yingziimage.oss-cn-beijing.aliyuncs.com/img/202209112131596.svg)]

5 - 预测

def predict_seq2seq(net, src_sentence, src_vocab, tgt_vocab, num_steps,device, save_attention_weights=False):
    """序列到序列模型的预测"""
    # 在预测时将net设置为评估模式
    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>']:
            break
        output_seq.append(pred)
    return ' '.join(tgt_vocab.to_tokens(output_seq)), attention_weight_seq

6 - 预测序列的评估

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):
        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}')
go . => va !, bleu 1.000
i lost . => j'ai perdu ., bleu 1.000
he's calm . => il est malade ., bleu 0.658
i'm home . => je vous suis <unk> ., bleu 0.000

7 - 小结

  • 根据“编码器-解码器”架构的设计,我们可以使⽤两个循环神经⽹络来设计⼀个序列到序列学习的模型
  • 在实现编码器和解码器时,我们可以使⽤多层循环神经⽹络
  • 我们可以使⽤遮蔽来过滤不相关的计算,例如在计算损失时
  • 在“编码器-解码器”训练中,强制教学⽅法将原始输出序列(⽽⾮预测结果)输⼊解码器
  • BLEU是⼀种常⽤的评估⽅法,它通过测量预测序列和标签序列之间的n元语法的匹配度来评估预测
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是使用PyTorch实现机器翻译的步骤: 1.读取和预处理数据 2.定义编码器和解码器模型 3.定义注意力机制 4.定义损失函数和优化器 5.训练模型 6.使用模型进行翻译 具体实现细节可以参考以下代码: ```python import os import zipfile import collections import time import torch import torch.nn as nn import torch.nn.functional as F from torch.utils import data from torch import optim from d2l.data.base import Vocab import d2l # 读取和预处理数据 def read_data_nmt(): data_dir = '/home/kesci/input/fraeng6506/fra-eng' with zipfile.ZipFile(os.path.join(data_dir, 'fra-eng.zip'), 'r') as f: raw_text = f.read('fra.txt').decode("utf-8") return raw_text raw_text = read_data_nmt() print(raw_text[:100]) def preprocess_nmt(text): text = text.replace('\u202f', ' ').replace('\xa0', ' ') no_space = lambda char, prev_char: ( True if char in (',', '!', '.') and prev_char != ' ' else False) out = [' '+char if i > 0 and no_space(char, text[i-1]) else char for i, char in enumerate(text.lower())] return ''.join(out) text = preprocess_nmt(raw_text) print(text[:100]) def tokenize_nmt(text, num_examples=None): source, target = [], [] for i, line in enumerate(text.split('\n')): if num_examples and i > num_examples: break parts = line.split('\t') if len(parts) == 2: source.append(parts[0].split(' ')) target.append(parts[1].split(' ')) return source, target source, target = tokenize_nmt(text) print(source[:3], target[:3]) # 建立词典 def build_vocab_nmt(tokens): tokens = [token for line in tokens for token in line] return Vocab(tokens, min_freq=3, use_special_tokens=True) src_vocab = build_vocab_nmt(source) print(list(src_vocab.token_to_idx.items())[:10]) tgt_vocab = build_vocab_nmt(target) print(list(tgt_vocab.token_to_idx.items())[:10]) # 将文本转换为数字序列 def encode_nmt(src_tokens, tgt_tokens, src_vocab, tgt_vocab): src_encoded = [[src_vocab[token] for token in line] for line in src_tokens] tgt_encoded = [[tgt_vocab[token] for token in line] for line in tgt_tokens] return src_encoded, tgt_encoded src_encoded, tgt_encoded = encode_nmt(source, target, src_vocab, tgt_vocab) print(src_encoded[:3], tgt_encoded[:3]) # 定义编码器和解码器模型 class Encoder(nn.Module): def __init__(self, vocab_size, embed_size, num_hiddens, num_layers, drop_prob=0): super(Encoder, self).__init__() self.embedding = nn.Embedding(vocab_size, embed_size) self.rnn = nn.LSTM(embed_size, num_hiddens, num_layers, dropout=drop_prob, bidirectional=True) def forward(self, inputs, state=None): # inputs shape: (batch_size, seq_len) # outputs shape: (seq_len, batch_size, 2*num_hiddens) embeddings = self.embedding(inputs) outputs, state = self.rnn(embeddings.permute([1, 0, 2]), state) return outputs.permute([1, 0, 2]), state class Decoder(nn.Module): def __init__(self, vocab_size, embed_size, num_hiddens, num_layers, attention_size, drop_prob=0): super(Decoder, self).__init__() self.embedding = nn.Embedding(vocab_size, embed_size) self.attention = Attention(num_hiddens, attention_size, drop_prob) self.rnn = nn.LSTM(num_hiddens + embed_size, num_hiddens, num_layers, dropout=drop_prob) self.out = nn.Linear(num_hiddens, vocab_size) def forward(self, cur_input, state, enc_outputs): # cur_input shape: (batch_size,) # state: the hidden state of the last time step # outputs shape: (batch_size, vocab_size) embeddings = self.embedding(cur_input).unsqueeze(0) context = self.attention(state[0][-1], enc_outputs) rnn_input = torch.cat([embeddings, context.unsqueeze(0)], dim=2) outputs, state = self.rnn(rnn_input, state) outputs = self.out(outputs).squeeze(0) return outputs, state class Attention(nn.Module): def __init__(self, enc_num_hiddens, dec_num_hiddens, attention_size, drop_prob=0): super(Attention, self).__init__() self.enc_attention = nn.Linear(enc_num_hiddens, attention_size, bias=False) self.dec_attention = nn.Linear(dec_num_hiddens, attention_size, bias=False) self.combined_attention = nn.Linear(attention_size, 1, bias=True) self.dropout = nn.Dropout(drop_prob) def forward(self, dec_state, enc_outputs): # dec_state shape: (batch_size, dec_num_hiddens) # enc_outputs shape: (batch_size, seq_len, enc_num_hiddens) dec_attention = self.dec_attention(dec_state).unsqueeze(1) enc_attention = self.enc_attention(enc_outputs) combined_attention = self.combined_attention(torch.tanh( enc_attention + dec_attention)) attention_weights = F.softmax(combined_attention.squeeze(2), dim=1) return torch.bmm(attention_weights.unsqueeze(1), enc_outputs).squeeze(1) # 定义损失函数和优化器 def sequence_mask(X, valid_len, value=0): maxlen = X.size(1) mask = torch.arange(maxlen)[None, :] < valid_len[:, None] X[~mask] = value return X class MaskedSoftmaxCELoss(nn.CrossEntropyLoss): def forward(self, pred, target, valid_len): weights = torch.ones_like(target) weights = sequence_mask(weights, valid_len).float() self.reduction = 'none' output = super(MaskedSoftmaxCELoss, self).forward(pred.transpose(1, 2), target) return (output * weights).mean(dim=1) def train_epoch_ch8(net, data_iter, lr, optimizer, device, use_random_iter): loss_sum, n = 0.0, 0 for batch in data_iter: optimizer.zero_grad() X, X_vlen, Y, Y_vlen = [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) # Teacher forcing Y_hat, _ = net(X, dec_input, X_vlen) loss = MaskedSoftmaxCELoss()(Y_hat, Y, Y_vlen) loss.sum().backward() d2l.grad_clipping(net, 1) num_tokens = Y_vlen.sum() optimizer.step() loss_sum += loss.sum().item() n += num_tokens.item() return loss_sum / n def train_ch8(net, train_iter, lr, num_epochs, device, use_random_iter=False): def init_weights(m): if type(m) == nn.Linear: nn.init.xavier_uniform_(m.weight) if type(m) == nn.LSTM: for param in m._flat_weights_names: if "weight" in param: nn.init.xavier_uniform_(m._parameters[param]) net.apply(init_weights) net.to(device) optimizer = torch.optim.Adam(net.parameters(), lr=lr) loss = MaskedSoftmaxCELoss() animator = d2l.Animator(xlabel='epoch', ylabel='loss', xlim=[1, num_epochs]) for epoch in range(num_epochs): timer = d2l.Timer() loss_avg = train_epoch_ch8(net, train_iter, lr, optimizer, device, use_random_iter) animator.add(epoch+1, loss_avg) print(f'epoch {epoch + 1}, loss {loss_avg:.3f}, ' f'time {timer.stop():.1f} sec') return net # 训练模型 embed_size, num_hiddens, num_layers = 64, 128, 2 attention_size, drop_prob, lr, batch_size, num_epochs = 10, 0.5, 0.01, 64, 300 train_iter = d2l.load_data_nmt(batch_size, num_examples=1000) encoder = Encoder(len(src_vocab), embed_size, num_hiddens, num_layers, drop_prob) decoder = Decoder(len(tgt_vocab), embed_size, num_hiddens, num_layers, attention_size, drop_prob) net = d2l.EncoderDecoder(encoder, decoder) device = torch.device('cuda' if torch.cuda.is_available() else 'cpu') net = train_ch8(net, train_iter, lr, num_epochs, device) # 使用模型进行翻译 def predict_ch8(net, src_sentence, src_vocab, tgt_vocab, num_steps, device): src_tokens = src_vocab[src_sentence.lower().split(' ')] 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.tensor(src_tokens, dtype=torch.long, device=device) enc_outputs, enc_state = net.encoder(enc_X.unsqueeze(0), enc_valid_len) dec_state = enc_state dec_X = torch.tensor([tgt_vocab['<bos>']], dtype=torch.long, device=device).reshape(1, 1) output_seq = [] for _ in range(num_steps): Y, dec_state = net.decoder(dec_X, dec_state, enc_outputs) dec_X = Y.argmax(dim=1).reshape(1, 1) pred = dec_X.squeeze(dim=0).type(torch.int32).item() if pred == tgt_vocab['<eos>']: break output_seq.append(pred) return ' '.join(tgt_vocab.to_tokens(output_seq)) src_sentence = 'They are watching.' print(predict_ch8(net, src_sentence, src_vocab, tgt_vocab, num_steps=10, device=device)) --相关问题--:
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值