用 NumPy 写一个RNN、LSTM,这可能是最好的入门方式!

2019 年 5 月 23 日 数说工作室

本文经机器之心(微信公众号:almosthuman2014)授权转载,禁二次转载

参与:思源



数说君导读:尽管 NumPy 不能利用 GPU 的并行计算能力,但利用它可以清晰了解底层的数值计算过程,这也许就是为什么 CS231n 等课程最开始都要求使用 NumPy手动实现深度网络的原因吧。




随着 TensorFlow 和 PyTorch 等框架的流行,很多时候搭建神经网络也就调用几行 API 的事。大多数开发者对底层运行机制,尤其是如何使用纯 NumPy 实现神经网络变得比较陌生。以前机器之心曾介绍过如何使用 NumPy 实现简单的卷积神经网络,但今天会介绍如何使用 NumPy 实现 LSTM 等循环神经网络。


一般使用纯 NumPy 实现深度网络会面临两大问题,首先对于前向传播,卷积和循环网络并不如全连接网络那样可以直观地实现。为了计算性能,实践代码与理论之间也有差别。其次,我们实现了前向传播后还需要继续实现反向传播,这就要求我们对矩阵微分和链式法则等数学基础都有比较充足的了解。


尽管 NumPy 不能利用 GPU 的并行计算能力,但利用它可以清晰了解底层的数值计算过程,这也许就是为什么 CS231n 等课程最开始都要求使用 NumPy 手动实现深度网络吧。


项目地址:https://github.com/krocki/dnc


在这个项目中,作者主要使用 NumPy 实现了 DNC、RNN 和 ,其中 RNN 代码借鉴了 A.Karpathy 以前写过的代码。此外,作者还写了 Gradient check 以确定实现的正确性,是不是感觉自深度学习框架流行以来,梯度检验这个词就渐渐消失了~


具体而言,这个项目是 DeepMind 于 2016 年发表在 Nature 的论文《Hybrid computing using a neural network with dynamic external memory》的实现,即可微神经计算机(DNC),其示例的任务是字符级预测。repo 中还包括 RNN(rnn-numpy.py) 和 LSTM (lstm-numpy.py) 的实现,一些外部数据(ptb, wiki)需要分别下载。


如下所示为 LSTM 的前向传播过程,Pyhon 2.7 的 xrange 改成 range 就好了 ˉ\(ツ)/ˉ:



 loss = 0

 # forward pass
 for t in xrange(len(inputs)):

 # encode in 1-of-k representation
 xs[t] = np.zeros((M, B))
 for b in range(0,B): xs[t][:,b][inputs[t][b]] = 1
 # gates, linear part
 gs[t] = np.dot(Wxh, xs[t]) + np.dot(Whh, hs[t-1]) + bh

 # gates nonlinear part
 #i, o, f gates
 gs[t][0:3*HN,:] = sigmoid(gs[t][0:3*HN,:])
 #c gate
 gs[t][3*HN:4*HN, :] = np.tanh(gs[t][3*HN:4*HN,:]) 

 #mem(t) = c gate * i gate + f gate * mem(t-1)
 cs[t] = gs[t][3*HN:4*HN,:] * gs[t][0:HN,:] + gs[t][2*HN:3*HN,:] * cs[t-1]
 # mem cell - nonlinearity
 cs[t] = np.tanh(cs[t])
 # new hidden state
 hs[t] = gs[t][HN:2*HN,:] * cs[t]
 # unnormalized log probabilities for next chars
 ys[t] = np.dot(Why, hs[t]) + by

 ###################
 mx = np.max(ys[t], axis=0)
 # normalize
 ys[t] -= mx 
 # probabilities for next chars
 ps[t] = np.exp(ys[t]) / np.sum(np.exp(ys[t]), axis=0

 for b in range(0,B):
 # softmax (cross-entropy loss)
 if ps[t][targets[t,b],b] > 0: loss += -np.log(ps[t][targets[t,b],b]) 


如上代码所示,最外层的循环 t 表示不同的时间步。而在每一个时间步下,首先需要计算不同的门控激活值,这三个门都是并在一起算的,这和我们在理论上看到的三个独立公式不太一样,但很合理。接下来按照 LSTM 单元的计算过程依次算出当前记忆内容 cs[t]、隐藏单元输出值 hs[t] 和最后的概率预测 ys[t]。最后只需要根据预测算损失值,并加入总体损失就行了。


除了上述的前向传播,更厉害的还是 RNN 和 LSTM 等的反向传播,即沿时间的反向传播(BPTT),这里就需要读者具体参考代码并测试了。


项目的使用


除了读源码外,当然我们也可以通过命令行直接试用模型效果,首先检验梯度等关键结构与代码:


python dnc-debug.py


下面的版本都是准备好的:


python rnn-numpy.py
python lstm-numpy.py
python dnc-numpy.py


该项目具有这些特点:数值计算仅依赖于 NumPy、添加了批处理、可将 RNN 修改为 LSTM,还能进行梯度检查。


该项目已经实现了 LSTM-控制器,2D 内存数组和内容可寻址的读/写。但有一个问题是,关键相似度的 softmax 会导致崩溃(除以 0),如果遇到这种情况,需要重新启动。该 repo 还有一些需要完成或改进的地方,包括动态内存分配和释放,实现更快、可保存的模型等。


在采样输出时,我们可以得到的数据包括时间、迭代次数、BPC(预测误差->每字符的位数,越低越好),以及处理速度(char/s)。


04163.009 s, iter 1048001.2808 BPC, 1488.38 char/s


如下展示了反向传播的数值梯度检验(最右边列的值应该小于 1e-4),中间列是计算得到的分析和数值梯度范围(这些应该或多或少都能匹配上)。


GRAD CHECK

Wxh: n = [-1.828500e-025.292866e-03] min 3.005175e-09, max 3.505012e-07
 a = [-1.828500e-025.292865e-03] mean 5.158434e-08 # 10/4
Whh: n = [-3.614049e-016.580141e-01] min 1.549311e-10, max 4.349188e-08
 a = [-3.614049e-016.580141e-01] mean 9.340821e-09 # 10/10
Why: n = [-9.868277e-027.518284e-02] min 2.378911e-09, max 1.901067e-05
 a = [-9.868276e-027.518284e-02] mean 1.978080e-06 # 10/10
Whr: n = [-3.652128e-021.372321e-01] min 5.520914e-09, max 6.750276e-07
 a = [-3.652128e-021.372321e-01] mean 1.299713e-07 # 10/10
Whv: n = [-1.065475e+004.634808e-01] min 6.701966e-11, max 1.462031e-08
 a = [-1.065475e+004.634808e-01] mean 4.161271e-09 # 10/10
Whw: n = [-1.677826e-011.803906e-01] min 5.559963e-10, max 1.096433e-07
 a = [-1.677826e-011.803906e-01] mean 2.434751e-08 # 10/10
Whe: n = [-2.791997e-021.487244e-02] min 3.806438e-08, max 8.633199e-06
 a = [-2.791997e-021.487244e-02] mean 1.085696e-06 # 10/10
Wrh: n = [-7.319636e-029.466716e-02] min 4.183225e-09, max 1.369062e-07
 a = [-7.319636e-029.466716e-02] mean 3.677372e-08 # 10/10
Wry: n = [-1.191088e-015.271329e-01] min 1.168224e-09, max 1.568242e-04
 a = [-1.191088e-015.271329e-01] mean 2.827306e-05 # 10/10
bh: n = [-1.363950e+009.144058e-01] min 2.473756e-10, max 5.217119e-08
 a = [-1.363950e+009.144058e-01] mean 7.066159e-09 # 10/10
by: n = [-5.594528e-025.814085e-01] min 1.604237e-09, max 1.017124e-05
 a = [-5.594528e-025.814085e-01] mean 1.026833e-06 # 10/10





推荐阅读:



登录查看更多
6

相关内容

长短期记忆网络(LSTM)是一种用于深度学习领域的人工回归神经网络(RNN)结构。与标准的前馈神经网络不同,LSTM具有反馈连接。它不仅可以处理单个数据点(如图像),还可以处理整个数据序列(如语音或视频)。例如,LSTM适用于未分段、连接的手写识别、语音识别、网络流量或IDSs(入侵检测系统)中的异常检测等任务。
一份循环神经网络RNNs简明教程,37页ppt
专知会员服务
172+阅读 · 2020年5月6日
神经网络的拓扑结构,TOPOLOGY OF DEEP NEURAL NETWORKS
专知会员服务
32+阅读 · 2020年4月15日
《深度学习》圣经花书的数学推导、原理与Python代码实现
【MIT深度学习课程】深度序列建模,Deep Sequence Modeling
专知会员服务
77+阅读 · 2020年2月3日
【书籍】深度学习框架:PyTorch入门与实践(附代码)
专知会员服务
163+阅读 · 2019年10月28日
【开源书】PyTorch深度学习起步,零基础入门(附pdf下载)
专知会员服务
110+阅读 · 2019年10月26日
PyTorch  深度学习新手入门指南
机器学习算法与Python学习
9+阅读 · 2019年9月16日
从零开始深度学习:利用numpy手写一个感知机
数萃大数据
10+阅读 · 2018年6月10日
一文读懂LSTM和循环神经网络
七月在线实验室
8+阅读 · 2018年4月18日
基础 | GRU神经网络
黑龙江大学自然语言处理实验室
27+阅读 · 2018年3月5日
小白都能看懂的神经网络入门,快收下吧~
码农翻身
3+阅读 · 2018年1月4日
RNN | RNN实践指南(1)
KingsGarden
21+阅读 · 2017年4月4日
Do RNN and LSTM have Long Memory?
Arxiv
19+阅读 · 2020年6月10日
Bivariate Beta LSTM
Arxiv
5+阅读 · 2019年10月7日
Arxiv
19+阅读 · 2018年10月25日
Relational recurrent neural networks
Arxiv
8+阅读 · 2018年6月28日
Arxiv
21+阅读 · 2018年5月23日
Arxiv
4+阅读 · 2017年7月25日
VIP会员
相关VIP内容
相关资讯
PyTorch  深度学习新手入门指南
机器学习算法与Python学习
9+阅读 · 2019年9月16日
从零开始深度学习:利用numpy手写一个感知机
数萃大数据
10+阅读 · 2018年6月10日
一文读懂LSTM和循环神经网络
七月在线实验室
8+阅读 · 2018年4月18日
基础 | GRU神经网络
黑龙江大学自然语言处理实验室
27+阅读 · 2018年3月5日
小白都能看懂的神经网络入门,快收下吧~
码农翻身
3+阅读 · 2018年1月4日
RNN | RNN实践指南(1)
KingsGarden
21+阅读 · 2017年4月4日
相关论文
Do RNN and LSTM have Long Memory?
Arxiv
19+阅读 · 2020年6月10日
Bivariate Beta LSTM
Arxiv
5+阅读 · 2019年10月7日
Arxiv
19+阅读 · 2018年10月25日
Relational recurrent neural networks
Arxiv
8+阅读 · 2018年6月28日
Arxiv
21+阅读 · 2018年5月23日
Arxiv
4+阅读 · 2017年7月25日
Top
微信扫码咨询专知VIP会员