发布时间:2023-12-02 09:00
在本系列文章中,我有跟大家分享过神经网络中两种经典层:卷积层(CNN) 和 全连接层(FC),这两种层的输入数据分别是:特征向量和图像(张量),在具体实现时输入的多样本之间是相互独立的,无联系关系。而且,卷积神经网络利用数据的局部相关性和权值共享的思想大大减少了网络的参数量,非常适合于图片这种具有 空间(Spatial) 局部相关性的数据,但自然界的信号除了具有空间维度之外,还有一个 时间(Temporal) 维度。具有时间维度的数据(也称作序列数据)有以下特点:
因此卷积神经网络并不擅长处理此类数据,本博客要介绍的循环神经网络可以较好地解决此类问题。
常见的序列数据有:语言、音乐、视频、股票、文字、DNA等等。
具有先后顺序的数据一般叫作 序列(Sequence),比如随时间而变化的商品价格数据就是非常典型的序列。考虑某件商品 A 在 1 月到 6 月之间的价格变化趋势,我们记为一维向量: [ x 1 , x 2 , x 3 , x 4 , x 5 , x 6 ] \\boldsymbol{[x_1,x_2,x_3,x_4,x_5,x_6]} [x1,x2,x3,x4,x5,x6],shape 为 [ 6 ] \\boldsymbol{[6]} [6]。如果要表示 b \\boldsymbol{b} b 件商品在 1 月到 6 月之间的价格变化趋势,可以记为 2 维张量:
[ [ x 1 ( 1 ) , x 2 ( 1 ) , ⋯ , x 6 ( 1 ) ] , [ x 1 ( 2 ) , x 2 ( 2 ) , ⋯ , x 6 ( 2 ) ] , ⋯ , [ x 1 ( 6 ) , x 2 ( 6 ) , ⋯ , x 6 ( 6 ) ] ] \\boldsymbol{\\left[[x_1^{(1)},x_2^{(1)},\\cdots,x_6^{(1)}],[x_1^{(2)},x_2^{(2)},\\cdots,x_6^{(2)}],\\cdots,[x_1^{(6)},x_2^{(6)},\\cdots,x_6^{(6)}]\\right]} [[x1(1),x2(1),⋯,x6(1)],[x1(2),x2(2),⋯,x6(2)],⋯,[x1(6),x2(6),⋯,x6(6)]]
其中 b \\boldsymbol{b} b 表示商品的数量,张量 shape 为 [ b , 6 ] \\boldsymbol{[b,6]} [b,6]。因此,表示一个序列信号只需要一个 shape 为 [ b , s ] \\boldsymbol{[b,s]} [b,s] 的张量,其中 b \\boldsymbol{b} b 为序列数量, s \\boldsymbol{s} s 为一个序列的长度。但是对于很多信号并不能直接用一个标量数值表示,比如每个时间戳产生长度为 n \\boldsymbol{n} n 的特征向量,则需要 shape 为 [ b , s , n ] \\boldsymbol{[b,s,n]} [b,s,n] 的张量才能表示。考虑更复杂的文本数据:句子。它在每个时间戳上面产生的单词是一个字符,并不是数值,不能直接用某个标量表示。
对于一个含有 n \\boldsymbol{n} n 个单词的句子,单词的一种简单表示方法就是 One-hot 编码。以英文句子为例,假设只考虑最常用的 1 万个单词,那么每个单词就可以表示为某位为 1,其它位置为 0 且长度为 1 万的稀疏 One-hot 向量;对于中文句子,如果也只考虑最常用的 5000 个汉字,同样的方法,一个汉字可以用长度为 5000 的 One-hot 向量表示。如果只考虑 n \\boldsymbol{n} n 个地名单词,可以将每个地名编码为长度为 n \\boldsymbol{n} n 的 One-hot 向量。
文字编码为数值的过程叫作 Word Embedding。One-hot 的编码方式实现 Word Embedding 简单直观,编码过程不需要学习和训练。在神经网络中,一般使用 词袋模型(Bag of words) 先对单词编号,如 2 表示 “I”,3 表示 “me” 等,然后再对每个进行 One-hot 编码,最后将每个词向量拼接成矩阵或者张量。但 One-hot 编码有以下缺点:
为了克服此不足,人们提出了另一种表示方法,即分布式表示。
首先要介绍一种相似度衡量方式。在自然语言处理领域,有专门的一个研究方向在探索如何学习到单词的表示向量(Word Vector),使得语义层面的相关性能够很好地通过 Word Vector 体现出来。一个衡量词向量之间相关度的方法就是 余弦相关度(Cosine similarity):
s i m i l a r i t y ( a , b ) ≜ c o s ( θ ) = a ⋅ b ∣ a ∣ ⋅ ∣ b ∣ \\boldsymbol{similarity(a,b)≜ cos(\\theta)=\\frac{a\\cdot b}{|a| \\cdot |b|}} similarity(a,b)≜cos(θ)=∣a∣⋅∣b∣a⋅b
其中 a \\boldsymbol{a} a 和 b \\boldsymbol{b} b 代表了两个词向量。
分布式表示最早由 Hinton 于 1986 年提出的,可以克服独热表示的缺点。 解决词汇与位置无关问题,可以通过计算向量之间的 距离(欧式距离、余弦距离等) 来体现词与词的相似性。其基本想法是直接用一个普通的向量表示 一个词,此向量为 [ 0.792 , − 0.177 , − 0.107 , 0.109 , − 0.542 , . . . ] \\boldsymbol{[0.792,-0.177,-0.107,0.109,-0.542,...]} [0.792,−0.177,−0.107,0.109,−0.542,...],常见维度为 50 或 100。用这种方式表示的向量,“麦克” 和 “话筒” 的距离会远远小于“ 麦克” 和 “天气” 的距离。
词向量的分布式表示的优点是解决了词汇与位置无关问题,不足是学习过程相对复杂且受训练语料的影响很大。训练这种向量表示的方法较多,常 见的有 LSA、PLSA、LDA、Word2Vec等,其中 Word2Vec 是 Google 在 2013 年开源的一个词向量计算工具,同时也是一套生成词向量的算法方案。
在神经网络中,单词的表示向量可以直接通过训练的方式得到,我们把单词的表示层叫作 Embedding 层。Embedding 层负责把单词编码为某个词向量 v \\boldsymbol{v} v,它接受的是采用数字编码的单词编号 i \\boldsymbol{i} i, 也就是词袋模型编码后的结果。系统总单词数量记为 N v o c a b \\boldsymbol{N_{vocab}} Nvocab,输出长度为 n \\boldsymbol{n} n 的向量 v \\boldsymbol{v} v:
v = f θ ( i ∣ N v o c a b , n ) \\boldsymbol{v = f_\\theta(i|N_{vocab}, n)} v=fθ(i∣Nvocab,n)
构建一个 shape 为 [ N v o c a b , n ] \\boldsymbol{[N_{vocab}, n]} [Nvocab,n] 的查询表对象 t a b l e \\boldsymbol{table} table,对于任意的单词编号 i \\boldsymbol{i} i,只需要查询到对应位置上的向量并返回即可:
v = t a b l e [ i ] \\boldsymbol{v = table[i]} v=table[i]
在 PyTorch 中,有一个 nn.Embedding(vocab_size,n) 类,它是 Module 类的子类,这里它接受最重要的两个初始化参数:词汇量大小,每个词汇向量表示的向量维度。Embedding 类返回的是一个形状为 [每句词个数,词维度] 的矩阵。
现在我们来考虑如何处理序列信号,以文本序列为例,考虑一个句子:
“ I h a t e t h i s b o r i n g m o v i e ” “I\\ hate\\ this\\ boring\\ movie” “I hate this boring movie”
通过 Embedding 层,可以将它转换为 shape 为 [ b , s , n ] \\boldsymbol{[b,s,n]} [b,s,n] 的张量, b \\boldsymbol{b} b 为句子数量, s \\boldsymbol{s} s 为句子长度, n \\boldsymbol{n} n 为词向量长度。上述句子可以表示为 shape 为 [ 1 , 5 , 10 ] \\boldsymbol{[1,5,10]} [1,5,10] 的张量,其中 5 代表句子单词长度,10 表示词向量长度。
接下来以情感分类任务为例来逐步探索能够处理序列信号的网络模型。情感分类任务通过分析给出的文本序列,提炼出文本数据表达的整体语义特征,从而预测输入文本的情感类型:正面评价 或者 负面评价。从分类角度来看,情感分类问题就是一个简单的二分类问题,与图片分类不一样的是,由于输入是文本序 列,传统的卷积神经网络并不能取得很好的效果。
对于每个词向量,分别使用一个全连接层网络来提取语义特征:
o = σ ( W t x t + b t ) \\boldsymbol{o = \\sigma(W_tx_t + b_t)} o=σ(Wtxt+bt)
上图中,各个单词的词向量通过 s \\boldsymbol{s} s 个全连接层分类网络 1 提取每个单词的特征,所有单词的特征最后合并,并通过分类网络 2 输出序列的类别概率分布,对于长度为 s \\boldsymbol{s} s 的句子来说,至少需要 s \\boldsymbol{s} s 个全网络层。但这种网络结构的缺点是:
针对全连接的第一个缺点,我们知道卷积神经网络之所以在处理局部相关数据时优于全连接网络,是因为它充分利用了权值共享的思想,大大减少了网络的参数量,使得网络训练起来更加高效。那在处理序列信号的问题上,我们可以借鉴权值共享的思想。
上图中,将这 s \\boldsymbol{s} s 个网络层参数共享,这样其实相当于使用一个全连接网络来提取所有单词的特征信息。通过权值共享后,参数量大大减少,网络训练变得更加稳定高效。但是,这种网络结构并没有考虑序列之间的先后顺序,将词向量打乱次序仍然能获得相同的输出,无法获取有效的全局语义信息。
针对全连接的第二个缺点,我们让网络能够提供一个单独的内存变量,每次提取词向量的特征并刷新内存变量,直至最后一个输入完成,此时的内存变量即存储了所有序列的语义特征,并且由于输入序列之间的先后顺序,使得内存变量内容与序列顺序紧密关联。
上图中,将内存变量实现为一个状态张量 h \\boldsymbol{h} h,除了原来的 W x h \\boldsymbol{W_{xh}} Wxh 参数共享外,这里额外增加了一个 W h h \\boldsymbol{W_{hh}} Whh 参数,每个时间戳 t \\boldsymbol{t} t 上状态张量刷新为:
h t = σ ( W x h x t + W h h h t − 1 + b ) \\boldsymbol{h_t = \\sigma(W_{xh}x_t+W_{hh}h_{t-1}+b) } ht=σ(Wxhxt+Whhht−1+b)
其中状态张量 h 0 \\boldsymbol{h_0} h0 为初始的内存状态,可以初始化为全 0, σ \\boldsymbol{\\sigma} σ 为激活函数,经过 s \\boldsymbol{s} s 个词向量的输入后得到网络最终的状态张量 h s \\boldsymbol{h_s} hs, h s \\boldsymbol{h_s} hs 较好地代表了句子的全局语义信息,基于 h s \\boldsymbol{h_s} hs 通过某个全连接层分类器即可完成情感分类任务。
经过上面的一步步分析探索,可以得到一种新型的网络结构,在每个时间戳 t \\boldsymbol{t} t,网络层接受当前时间戳的输入 x t \\boldsymbol{x_t} xt 和上一个时间戳的网络状态向量 h t − 1 \\boldsymbol{h_{t-1}} ht−1 ,经过:
h t = f θ ( h t − 1 , x t ) \\boldsymbol{h_t = f_\\theta(h_{t-1},x_t)} ht=fθ(ht−1,xt)
变换后得到当前时间戳的新状态向量 h t \\boldsymbol{h_t} ht,并写入内存状态中,其中 f θ \\boldsymbol{f_\\theta} fθ 代表了网络的运算逻辑, θ \\boldsymbol{\\theta} θ 为网络参数集。如果在每个时间戳上,网络层均有输出产生 o t \\boldsymbol{o_t} ot, o t = g γ ( h t ) \\boldsymbol{o_t = g_\\gamma(h_t)} ot=gγ(ht),即将网络的状态向量变换后输出。
网络循环接受序列的每个特征向量 x t \\boldsymbol{x_t} xt,并刷新内部状态向量 h t \\boldsymbol{h_t} ht,同时形成输出 o t \\boldsymbol{o_t} ot。对于这种网络结构,我们把它叫做 循环神经网络(Recurrent Neural Network,简称 RNN)。
如果使用张量 W x h \\boldsymbol{W_{xh}} Wxh、 W h h \\boldsymbol{W_{hh}} Whh 和偏置 b \\boldsymbol{b} b 来参数化 f θ \\boldsymbol{f_\\theta} fθ 网络,并按照如下方式更新内存状态:
h t = σ ( W x h x t + W h h h t − 1 + b ) \\boldsymbol{h_t = \\sigma(W_{xh}x_t+W_{hh}h_{t-1}+b) } ht=σ(Wxhxt+Whhht−1+b)
我们把这种网络叫做基本的循环神经网络,如无特别说明,一般说的循环神经网络即指这种实现。在循环神经网络中,激活函数更多地采用 tanh 函数。并且可以选择不使用偏执 b \\boldsymbol{b} b 来进一步减少参数量。状态向量 h t \\boldsymbol{h_t} ht 可以直接用作输出,即 o t = h t \\boldsymbol{o_t = h_t} ot=ht,也可以对 h t \\boldsymbol{h_t} ht 做一个简单的线性变换 o t = W h o h t \\boldsymbol{o_t = W_{ho}h_t} ot=Whoht 后得到每个时间戳上的网络输出 o t \\boldsymbol{o_t} ot。
经过上面的推到,我们知道 RNN 是怎么来的的,内部的公式如何计算的。由上面再次总结 RNN 的核心公式如下:
要注意的是 RNN 中的激活函数可以是 ReLU、sigmoid、tanh,不过一般来说使用 tanh 较多。
循环神经网络适合于处理序列数据,序列长度一般不固定,比如我们在前面举的文本分类的是有多个输入,只有一个输出;在讲解 RNN 结构时每一个输入都有一个输出,因此,RNN 应用非常广泛。我们可以根据应用场景将 RNN 分成以下几种类型:
上图中每一个矩形是一个向量,箭头则表示函数(比如矩阵相乘)。 其中最下层为输入向量,最上层为输出向量,中间层表示 RNN 的状态。从左到右:
此外,RNN 也可以堆叠几层,如下:
计算公式跟一层 RNN 完全一样,只是输出上有些区别,这里不展开讲述。
这篇博客就不跟大家探讨 RNN 的梯度传播了,只是探讨一下传播过程中的问题。RNN 跟别的网络结构一样,在反向传播中依然存在梯度爆炸和梯度消失的问题,主要的解决方法如下:
下一篇:循环神经网络:LSTM、GRU