当前位置:   article > 正文

深度学习(PyTorch)——循环神经网络(RNN)进阶篇_pytorch rnn

pytorch rnn

B站up主“刘二大人”视频 笔记 

一、循环神经网络的背景

传统神经网络(包括CNN),输入和输出都是互相独立的。图像上的猫和狗是分隔开的,但有些任务,后续的输出和之前的内容是相关的。例如:我是中国人,我的母语是____。这是一道填空题,需要依赖于之前的输入。

所以,RNN引入“记忆”的概念,也就是输出需要依赖于之前的输入序列,并把关键输入记住。循环2字来源于其每个元素都执行相同的任务。

它并⾮刚性地记忆所有固定⻓度的序列,而是通过隐藏状态来存储之前时间步的信息。

前向神经网络和 CNN 在很多任务中都取得不错的效果,但是这些网络结构的通常比较适合用于一些不具有时间或者序列依赖性的数据,接受的输入通常与上一时刻的输入没有关系。

但是序列数据不同,输入之间存在着先后顺序,当前输入的结果通常与前后的输入都有关。例如一段句子包含 4 个输入单词 :“我”、“去”、“商场”、“打车”,4 个单词通过不同的顺序排列,会有不同的意思,“我打车去商场” 和 “我去商场打车”。因此我们通常需要按照一定的顺序阅读句子才能理解句子的意思。

面对这种情况我们就需要用到循环神经网络了,循环神经网络按照顺序处理所有的输入,每一时刻 t,都会存在一个向量 h 保存与 t 时刻相关的信息 (可以是 t 时刻前的信息或者 t 时刻后的信息)。通过向量 h 与输入向量 x,就可以比较准确地判断当前的结果。在下文中的符号表示:

xt 表示 t 时刻的输入向量(例如第 t 个单词的词向量)。
ht 表示 t 时刻的隐藏向量 (包含了从开始一直到 t 时刻的相关信息)。
yt 表示 t 时刻的输出向量 (通常是预测的结果)。

二、循环神经网络训练时出现的问题

由于RNN特有的memory会影响后期其他的RNN的特点,梯度时大时小,learning rate没法个性化的调整,导致RNN在train的过程中,Loss会震荡起伏,为了解决RNN的这个问题,在训练的时候,可以设置临界值,当梯度大于某个临界值,直接截断,用这个临界值作为梯度的大小,防止大幅震荡。

程序如下:

  1. import csv
  2. import gzip
  3. import math
  4. import time
  5. import torch
  6. from torch.nn.utils.rnn import pack_padded_sequence
  7. from torch.utils.data import Dataset, DataLoader
  8. import matplotlib.pyplot as plt
  9. import numpy as np
  10. HIDDEN_SIZE = 100 # GRU输出隐层的维度
  11. BATCH_SIZE = 256
  12. N_LAYER = 2 # 用2层的GRU
  13. N_EPOCHS = 100
  14. N_CHARS = 128 # 字母表的大小 ASCII表
  15. USE_GPU = False
  16. class NameDataset(Dataset):
  17. def __init__(self,is_train_set=True):
  18. filename = 'names_train.csv.gz' if is_train_set else 'names_test.csv.gz'
  19. with gzip.open(filename,'rt') as f:
  20. reader = csv.reader(f)
  21. rows = list(reader) # 得到数据的(name,language)
  22. self.names = [row[0] for row in rows] # 得到数据的name
  23. self.len = len(self.names)
  24. self.countries = [row[1] for row in rows] # 得到数据的language
  25. self.country_list = list(sorted(set(self.countries)))
  26. # set把列表变成集合,去除重复元素,然后用sorted排序,最后变成列表
  27. self.country_dict = self.getCountryDict() # 把列表变成词典
  28. self.country_num = len(self.country_list)
  29. # country和index相当于字典的键值对
  30. def __getitem__(self, index): # 获得对应得索引index
  31. return self.names[index],self.country_dict[self.countries[index]]
  32. #
  33. def __len__(self): # 返回数据集长度
  34. return self.len
  35. def getCountryDict(self): # 构造country与index的字典
  36. country_dict = dict()
  37. for idx,country_name in enumerate(self.country_list,0):
  38. country_dict[country_name] = idx
  39. return country_dict
  40. def idx2country(self,index): # 根据索引index返回国家的字符串
  41. return self.country_list[index]
  42. def getCountriesNum(self): # 返回国家的数量
  43. return self.country_num
  44. trainset = NameDataset(is_train_set=True)
  45. trainloader = DataLoader(trainset,batch_size=BATCH_SIZE,shuffle=True)
  46. testset = NameDataset(is_train_set=False)
  47. testloader = DataLoader(testset,batch_size=BATCH_SIZE,shuffle=False)
  48. N_COUNTRY = trainset.getCountriesNum() # 多少分类
  49. def create_tensor(tensor): # 判断是否使用GPU
  50. if USE_GPU:
  51. device = torch.device("cuda:0")
  52. tensor = tensor.to(device)
  53. return tensor
  54. class RNNClassifier(torch.nn.Module):
  55. def __init__(self,input_size,hidden_size,output_size,n_layers=1,bidirectional=True):
  56. super(RNNClassifier, self).__init__()
  57. self.hidden_size = hidden_size
  58. self.n_layers = n_layers
  59. self.n_directions = 2 if bidirectional else 1
  60. self.embedding = torch.nn.Embedding(input_size,hidden_size)
  61. # 输入emb层的维度(seqlen,batchsize),输出emb层的维度(seqlen,batchsize,hiddensize)
  62. self.gru = torch.nn.GRU(hidden_size,hidden_size,n_layers,bidirectional=bidirectional)
  63. #输入GRU维度input:(seqlen,batchsize,hiddensize),hidden:(nlayers*nDirections,batchsize,hiddensize)
  64. #输出GRU维度output:(seqlen,batchsize,hiddensize*nDirections),hidden:(nlayers*nDirections,batchsize,hiddensize)
  65. self.fc = torch.nn.Linear(hidden_size * self.n_directions,output_size)
  66. def _init_hidden(self,batch_size): # 创建一个全0的初始隐层
  67. hidden = torch.zeros(self.n_layers*self.n_directions,batch_size,self.hidden_size)
  68. return create_tensor(hidden)
  69. def forward(self,input,seq_lengths):
  70. input = input.t() # 矩阵的装置 batch*seqlen->seqlen*batch
  71. batch_size = input.size(1)
  72. hidden = self._init_hidden(batch_size)
  73. embedding = self.embedding(input) # 输出的维度(seqlen,batchsize,hiddensize)
  74. gru_input = pack_padded_sequence(embedding,seq_lengths)
  75. output,hidden = self.gru(gru_input,hidden)
  76. if self.n_directions == 2:
  77. hidden_cat = torch.cat([hidden[-1],hidden[-2]],dim=1)
  78. else:
  79. hidden_cat = hidden[-1]
  80. fc_output = self.fc(hidden_cat)
  81. return fc_output
  82. def time_since(since):
  83. s = time.time() - since # 当前时间-开始时间
  84. m = math.floor(s/60) # 变成分钟
  85. s -= m*60
  86. return '%dm %ds' % (m,s) # 返回多少分钟和多少秒
  87. def name2list(name): # 把每一个名字变成列表
  88. arr = [ord(c) for c in name]
  89. return arr,len(arr)
  90. def make_tensors(names,countries):
  91. sequences_and_lengths = [name2list(name) for name in names]
  92. name_sequences = [sl[0] for sl in sequences_and_lengths]
  93. seq_lengths = torch.LongTensor([sl[1] for sl in sequences_and_lengths])
  94. countries = countries.long()
  95. # padding 0
  96. seq_tensor = torch.zeros(len(name_sequences),seq_lengths.max()).long()
  97. for idx,(seq,seq_len) in enumerate(zip(name_sequences,seq_lengths),0):
  98. seq_tensor[idx,:seq_len] = torch.LongTensor(seq)
  99. # 按照序列长度进行排序,最长的序列放前面
  100. seq_lengths,perm_idx = seq_lengths.sort(dim=0,descending=True)
  101. seq_tensor = seq_tensor[perm_idx]
  102. countries = countries[perm_idx]
  103. return create_tensor(seq_tensor),\
  104. create_tensor(seq_lengths),\
  105. create_tensor(countries)
  106. def trainModel():
  107. total_loss = 0
  108. for i,(names,countries) in enumerate(trainloader,1):
  109. inputs,seq_lengths,target = make_tensors(names,countries)
  110. output = classifier(inputs,seq_lengths) # 分类器
  111. loss = criterion(output,target) # 计算损失
  112. optimizer.zero_grad() # 清零
  113. loss.backward() # 反馈
  114. optimizer.step() # 更新
  115. total_loss += loss.item()
  116. if i % 10 == 0:
  117. print(f'[{time_since(start)}] Epoch {epoch}',end='')
  118. print(f'[{i*len(inputs)}/{len(trainset)}]', end='')
  119. print(f'loss={total_loss / (i*len(inputs))}')
  120. return total_loss
  121. def testModel():
  122. correct = 0
  123. total = len(testset)
  124. print("evaluating trained model ...")
  125. with torch.no_grad():
  126. for i,(names,countries) in enumerate(testloader,1):
  127. inputs,seq_lengths,target = make_tensors(names,countries)
  128. output = classifier(inputs,seq_lengths)
  129. pred = output.max(dim=1,keepdim=True)[1]
  130. correct += pred.eq(target.view_as(pred)).sum().item()
  131. percent = '%.2f' % (100*correct / total)
  132. print(f'Test set:Accuracy {correct}/{total} {percent}%')
  133. return correct / total
  134. if __name__ == '__main__':
  135. classifier = RNNClassifier(N_CHARS,HIDDEN_SIZE,N_COUNTRY,N_LAYER)
  136. if USE_GPU:
  137. device = torch.device("cuda:0")
  138. classifier.to(device)
  139. criterion = torch.nn.CrossEntropyLoss() # 构造损失函数
  140. optimizer = torch.optim.Adam(classifier.parameters(),lr=0.001) # 构造优化器
  141. start = time.time()
  142. print("Training for %d epochs..." % N_EPOCHS)
  143. acc_list=[]
  144. for epoch in range(1,N_EPOCHS+1):
  145. trainModel()
  146. acc = testModel()
  147. acc_list.append(acc)
  148. epoch = np.arange(1,len(acc_list) + 1,1)
  149. acc_list = np.array(acc_list)
  150. plt.plot(epoch,acc_list)
  151. plt.xlabel("Epoch")
  152. plt.ylabel('Accuracy')
  153. plt.grid()
  154. plt.show()

运行结果如下:

 

视频截图如下: 

 

 

 

 

 

 

 

 

 

 

 

 ​​​​​​​

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/神奇cpp/article/detail/897180
推荐阅读
相关标签
  

闽ICP备14008679号