【深度学习入门到进阶】必看系列,含激活函数、优化策略、损失函数、模型调优、归一化算法、卷积模型、序列模型、预训练模型、对抗神经网络等
专栏详细介绍:【深度学习入门到进阶】必看系列,含激活函数、优化策略、损失函数、模型调优、归一化算法、卷积模型、序列模型、预训练模型、对抗神经网络等
本专栏主要方便入门同学快速掌握相关知识。后续会持续把深度学习涉及知识原理分析给大家,让大家在项目实操的同时也能知识储备,知其然、知其所以然、知何由以知其所以然。
声明:部分项目为网络经典项目方便大家快速学习,后续会不断增添实战环节(比赛、论文、现实应用等)
专栏订阅:深度学习入门到进阶专栏
自回归模型(Autoregressive Model, AR),通过估计一串文本序列的生成概率分布进行建模。一般而言,AR模型通过要么从前到后计算文本序列概率,要么从后向前计算文本序列概率,但不论哪种方式的建模,都是单向的。即在预测一个单词的时候无法同时看到该单词位置两边的信息。假设给定的文本序列 x = ( x 1 , x 2 , . . . , x n ) x=(x_1, x_2, ..., x_n) x=(x1,x2,...,xn),其从左到右的序列生成概率为:
p ( x ) = ∏ t = 1 n p ( x t ∣ x < t ) p(x)=\prod_{t=1}^n p(x_t|x_{
自编码模型(Autoencoding Model, AE), 通过从破坏的输入文本序列中重建原始数据进行建模。例如BERT通过预测【mask】位置的词重建原始序列。它的优点在于在预测单词的时候能够同时捕获该单词位置前后双向的信息;它的缺点是预训练过程中采用了mask单词的策略,然而微调阶段并没有,因此导致了预训练阶段和微调阶段的的GAP,另外在训练过程中,对不同mask单词的预测是相互独立的。假设序列中被mask的词为 w ∈ W m w\in W_m w∈Wm,未被mask的词为 w ∈ W n w\in W_n w∈Wn,则其相应的计算概率为:
p ( x ) = ∏ w ∈ W m p ( w ∣ W n ) p(x) = \prod_{w\in Wm} p(w|W_n) p(x)=w∈Wm∏p(w∣Wn)
这里对传统的AR和AE模型简单总结一下,AR模型是生成式模型,是单向的;AE模型是判别式模型,是双向的。鉴于传统的AR模型和AE模型自身的优点和缺点,XLNet期望能够融合两者的优点同时又避免两者的缺点,这就是XLNet的设计思路。
整体上XLNet是基于AR模型的建模思路设计的,同时避免了只能单向建模的缺点,因此它是一种能看得见双向信息的广义AR模型。作为一个AR模型,XLNet并没有采用预测mask单词的方式进行建模,因此它不存在上述预训练-微调的GAP,更不存在预测mask的独立性假设。
Permuatation Language Mode (下文将简称为PLM) 是XLNet的核心建模思路,在正式介绍之前,我们再来回顾一下AR模型的建模策略,给定一串文本序列 x = [ x 1 , x 2 , . . . , x n ] \text{x}=[x_1,x_2,...,x_n] x=[x1,x2,...,xn],其中每个 x i x_i xi表示一个token,AR模型的通过最大化下边这个似然函数进行建模:
m a x θ l o g P θ ( x ) = ∑ t = 1 n l o g p θ ( x t ∣ x < t ) = ∑ t = 1 n l o g e x p ( h θ ( x 1 : t − 1 ) T e ( x t ) ) ∑ x ′ e x p ( h θ ( x 1 : t − 1 ) T e ( x ′ ) ) \begin{split} \begin{align} \mathop{max}\limits_{\theta} \quad log \; P_{\theta}(\text{x}) &= \sum_{t=1}^{n}log\;p_{\theta}(x_t|x_{
这里, x < t \text{x}_{
这种建模方式是单向的,为了在预测某个位置单词的时候,能够让模型看见双向的信息,XLNet采用了全排列的思路,允许模型在不同文本序列排列上进行建模,但模型的参数在不同序列上是共享的,相当于是模型能够看见预测位置单词左右两侧的信息。
举个例子,假设当前有文本序列 x = [ x 1 , x 2 , x 3 ] \text{x}=[x_1,x_2,x_3] x=[x1,x2,x3],这串序列中共有3个token,这三个token共计有6种排列组合方式,其相关的索引序列为:
z 1 = ( 1 , 2 , 3 ) \text{z}_1 = (1,2,3) z1=(1,2,3)
z 2 = ( 1 , 3 , 2 ) \text{z}_2=(1,3,2) z2=(1,3,2)
z 3 = ( 2 , 1 , 3 ) \text{z}_3=(2,1,3) z3=(2,1,3)
z 4 = ( 2 , 3 , 1 ) \text{z}_4=(2,3,1) z4=(2,3,1)
z 5 = ( 3 , 1 , 2 ) \text{z}_5=(3,1,2) z5=(3,1,2)
z 6 = ( 3 , 2 , 1 ) \text{z}_6=(3,2,1) z6=(3,2,1)
采用索引序列 z 1 \text{z}_1 z1的文本序列为 x = [ x 1 , x 2 , x 3 ] \text{x}=[x_1, x_2, x_3] x=[x1,x2,x3],采用索引序列 z 2 \text{z}_2 z2的文本序列为 x = [ x 1 , x 3 , x 2 ] \text{x}=[x_1, x_3, x_2] x=[x1,x3,x2],如果模型在训练过程中能同时看到这样的两个排列,那么在预测 x 2 x_2 x2的时候将可以看到 x 1 x_1 x1,又可以看到 x 3 x_3 x3,因为训练过程中模型的参数是共享的,因此相当于模型能够看到 x 2 x_2 x2前后双向的信息。
下面正式归纳一下XLNet的目标函数,假设给定一串序列 x = [ x 1 , x 2 , x 3 , . . . , x n ] \text{x}=[x_1,x_2,x_3,...,x_n] x=[x1,x2,x3,...,xn],它将有 n ! n! n!个不同的排列组合 Z = [ z 1 , z 2 , . . . , z n ! ] \mathbb{Z}=[\text{z}_1,\text{z}_2,...,\text{z}_{n!}] Z=[z1,z2,...,zn!],令 z ∈ Z \text{z}\in \mathbb{Z} z∈Z表示某一排列方式, z t z_{t} zt表示为 z \text{z} z这个排列的第 t t t个位置, z < t \text{z}_{
m a x θ L = E z ∼ Z [ ∑ t = 1 n l o g p θ ( x z t ∣ x z < t ) ] = ∑ z ∼ Z p ( z ) [ ∑ t = 1 n l o g p θ ( x z t ∣ x z < t ) ] \begin{split} \begin{align} \mathop{max}_{\theta} \quad L &=\mathbb{E}_{\text{z}∼\mathbb{Z}} \left[ \sum_{t=1}^n log\;p_{\theta}(x_{z_t}|\text{x}_{\text{z}_{
但是一个长度为 n n n的文本序列,其排列组合数为 n ! n! n!,数量实在庞大,不便训练,所以在实际训练过程中,XLNet是通过采样的方式逼近目标函数的期望,即:
m a x θ L = E z ∼ Z [ ∑ t = 1 n l o g p θ ( x z t ∣ x z < t ) ] ≈ 1 m ∑ i = 1 m ∑ t = 1 n l o g p θ ( x z i t ∣ x z i < t ) \begin{split} \begin{align} \mathop{max}_{\theta} \quad L &=\mathbb{E}_{\text{z}∼\mathbb{Z}} \left[ \sum_{t=1}^n log\;p_{\theta}(x_{z_t}|\text{x}_{z
其中, z i t z_{it} zit表示第 i i i个排列的第 t t t个位置, x z i < t \text{x}_{\text{z}_i
图1 不同排列计算第3个单词输出的示意图
图1中 m e m ( 0 ) mem^{(0)} mem(0)和 m e m ( 1 ) mem^{(1)} mem(1)代表前一个的segment的缓存信息,另外,输入的序列顺序都是固定的自然语序,position embedding还是按照正常的顺序确定的。只是不同的排列中,参与分解计算的内容不同。具体来讲,对第一个分解次序 3 → 2 → 4 → 1 3\rightarrow2\rightarrow4\rightarrow1 3→2→4→1,因为 x 3 x_3 x3位于最前边,所以在这个分解次序中看不到其他的token内容,只能看到前一个segment的缓存信息;对第一个分解次序 2 → 4 → 3 → 1 2\rightarrow4\rightarrow3\rightarrow1 2→4→3→1, x 3 x_3 x3前边有 x 2 x_2 x2和 x 4 x_4 x4,所以在计算 x 3 x_3 x3位置输出的时候使用了 x 2 x_2 x2和 x 4 x_4 x4。
这个想法就是PLM的建模思路,看到这里,相信你已经很好地理解了。
上边看似找到了一个比较好想法去让AR模型在预测一个单词的时候同时能够看到前后双向的信息,但具体怎么来建模实现呢?使用原始的transformer技术直接实现可能会带来一些问题,具体来说,假设当前有两个排列 z 1 \text{z}^{1} z1和 z 2 \text{z}^2 z2,他们之间具有如下的关系:
z < t 1 = z < t 2 = z < t b u t z t 1 = i ≠ j = z t 2 \text{z}_{
这种情况下,使用经典transformer的方式去预测这两个排列 z t z_t zt位置的输出,将会有:
p θ ( X i = x ∣ x z < t ) ⏟ z t 1 = i , z < t 1 = z < t = p θ ( X j = x ∣ x z < t ) ⏟ z t 2 = i , z < t 2 = z < t = e x p ( e ( x ) T h ( x z < t ) ) ∑ x ′ e x p ( e ( x ′ ) T h ( x z < t ) ) \underbrace{p_{\theta}(X_{i}=x|\text{x}_{\text{z}_{
显然在这种情况下,预测第 i i i个位置的单词和预测第 j j j个位置的单词的概率分布是相同的,这肯定是不对的,因此使用经典的transformer是无法去做Permutation Language Model建模的。
为了解决这个问题,XLNet在预测目标 z t z_t zt位置的token时,向其引入了位置信息 z t z_t zt,重新定义的预测token概率分布的计算方式为:
p θ ( x z t ∣ x z < t ) = e x p ( e ( x ) T g θ ( x z < t , z t ) ) ∑ x ′ e x p ( e ( x ′ ) T g θ ( x z < t , z t ) ) ) p_{\theta}(x_{z_t}|\text{x}_{\text{z}_{
从公式中可以看到,其在预测 z t z_t zt位置token的时候,引入了位置信息 z t z_t zt。这样就能解决上述的问题,即经过这个变换后上式将变为:
p θ ( X i = x ∣ x z < t ) ⏟ z t 1 = i , z < t 1 = z < t = e x p ( e ( x ) T g θ ( x z < t , i ) ) ∑ x ′ e x p ( e ( x ′ ) T g θ ( x z < t , i ) ) ) ≠ p θ ( X j = x ∣ x z < t ) ⏟ z t 2 = i , z < t 2 = z < t = e x p ( e ( x ) T g θ ( x z < t , j ) ) ∑ x ′ e x p ( e ( x ′ ) T g θ ( x z < t , j ) ) ) \underbrace{p_{\theta}(X_{i}=x|\text{x}_{\text{z}_{
从上边讨论的这些可以看到,当预测 z t z_t zt位置的token时,最多只能使用位置信息 z t z_t zt,而不能使用该 z t z_t zt对应的内容 x z t x_{z_t} xzt,否则,就相当于使用 x z t x_{z_t} xzt来预测 x z t x_{z_t} xzt自己,这没有什么意义;当预测 j > t j>t j>t后的token x z j x_{z_j} xzj时,不仅需要位置信息 z t z_t zt,同时还需要该位置对应的内容 x z t x_{z_t} xzt。
然而,经典的transformer中是将两者信息在输入层相加融合之后进行后续计算的,因此XLNet提出了一种双流自注意力机制:content-stream 和 query stream,下面我们将来具体探讨一下它们。
content-stream 提供了内容方面的表达 content representation h θ ( x z ≤ t ) h_{\theta}(\text{x}_{\text{z}_{\leq t}} ) hθ(xz≤t),简记为 h z t h_{z_t} hzt,它等同经典的transformer 的状态向量,这个向量中既包含了位置信息 z t z_t zt,又包含了内容信息 x z t x_{z_t} xzt。
query-stream 提供了查询方面的表达 query representation g θ ( x z < t , z t ) g_{\theta}(\text{x}_{\text{z}_{
图2 双流机制计算图
图2展示了分解顺序为 3 → 2 → 4 → 1 3 \rightarrow 2 \rightarrow 4 \rightarrow 1 3→2→4→1的two-stream计算过程,我们通过这张图来具体聊聊如何去定义 g θ ( x z < t , z t ) g_{\theta}(\text{x}_{\text{z}_{
图2a展示了content-stream的自注意力计算,其中 h i ( 0 ) h_i^{(0)} hi(0)是由token的embedding进行初始化,可以看到它的计算和经典的transormer是一致的,因为在分解顺序中1位于最后,因此它能看见前边所有的token内容,最终计算得出的 h 1 ( 1 ) h_1^{(1)} h1(1)同时包含了第 1 1 1个位置的内容信息。
图2b展示了query-stream的自注意力计算,其中 g i ( 0 ) g_i^{(0)} gi(0)由可学习的参数进行初始化,因为它能看见token 3,2,4的内容信息,所以这里用的是内容信息 h 3 ( 0 ) , h 2 ( 0 ) , h 4 ( 0 ) h_3^{(0)},h_2^{(0)},h_4^{(0)} h3(0),h2(0),h4(0),同时对于第1个位置,只能使用位置信息,而不能使用内容信息,所以这里使用的是 g 1 ( 0 ) g_1^{(0)} g1(0),它并不包含第1个位置的内容信息。 这样就能比较好地建模 g θ ( x z < t , z t ) g_{\theta}(\text{x}_{\text{z}_{
图2c展示了整体的计算流程,最后一层输出的query向量就是我们想要的 g θ ( x z < t , z t ) g_{\theta}(\text{x}_{\text{z}_{
g z t ( m ) ← Attention ( Q = g z t ( m − 1 ) , K V = h z < t ( m − 1 ) ; θ ) , (query stream: use z t but cannot see x z t ) h z t ( m ) ← Attention ( Q = h z t ( m − 1 ) , K V = h z ≤ t ( m − 1 ) ; θ ) , (content stream: use both z t and x z t ) \begin{split} \begin{align} g_{z_t}^{(m)} & \leftarrow \text{Attention}(Q=g_{z_t}^{(m-1)},\, KV=h_{\text{z}_{
以上是XLNet在预训练阶段的计算过程,这里需要注意的是,在微调阶段,XLNet仅仅使用content respresentation进行fine-tune下游任务。
由于XLNet本身是个AR模型,它可以完美融入Transformer-XL的思想:相对位置编码和segment循环机制。这两者的原理部分感兴趣的同学可以去阅读Transformer-XL内容,本文重点讨论一下segment循环机制向XLNet的融入过程。
顾名思义,segment循环机制是指长序列切分成 n n n个segment (文本片段),然后将每个segment依次传入模型之中,同时传入到模型中,同时传入到模型中还有上一个segment的产生的输出,这个操作有点像RNN,接收上一步的输出和当前步骤的输入,然后根据两者计算产生当前步骤的输出,只不过RNN的循环单位是单词,XLNet的循环单位是segment。
给定一个长序列 s \text{s} s,上一个segment为 x ~ = s 1 : n \tilde{\text{x}}=s_{1:n} x~=s1:n,其对应的排列用 z ~ \tilde{\text{z}} z~表示;当前的segment为 x = s n + 1 : 2 n \text{x}=s_{n+1:2n} x=sn+1:2n,其对应的排列用 z \text{z} z表示。基于排列 z ~ \tilde{\text{z}} z~处理第1个segment,并将其输出进行缓存,第 m m m层的输出用 h ( m ) ~ \tilde{h^{(m)}} h(m)~表示。则第2个segment的计算可以按照如下方式进行:
g z t ( m ) ← Attention ( Q = g z t ( m − 1 ) , K V = [ h ~ ( m − 1 ) , h z < t ( m − 1 ) ] ; θ ) , (query stream: use z t but cannot see x z t ) h z t ( m ) ← Attention ( Q = h z t ( m − 1 ) , K V = [ h ~ ( m − 1 ) , h z ≤ t ( m − 1 ) ] ; θ ) , (content stream: use both z t and x z t ) \begin{split} \begin{align} g_{z_t}^{(m)} & \leftarrow \text{Attention}(Q=g_{z_t}^{(m-1)},\, KV=[\tilde{h}^{(m-1)},h_{\text{z}_{
即将前一个segment的输出和当前位置 z t z_t zt能看到的内容进行拼接,然后进行Self-Attention融合计算。
这里需要注意的是,由于序列中的position embedding 仅仅依赖于原始序列(输入序列)的位置,而不是排列的顺序,所以一旦前一个segment的输出 h ~ ( m ) \tilde{h}^{(m)} h~(m)确定,上述的Attention计算和前一个segment的分解顺序无关。这允许我们去缓存或者复用前一个segment的输出,而不用去管前一个segment的分解顺序。
最开始的时候有提到,AR模型通过估计一串文本序列的生成概率分布进行建模: ∑ t = 1 n l o g p θ ( x z t ∣ x z < t ) \sum_{t=1}^n log\;p_{\theta}(x_{z_t}|\text{x}_{\text{z}
因此XLNet在训练过程中,只选择预测序列最后面的部分位置的token,这里涉及到一个切分点位置 c c c,它将指示不预测在 c c c前边的位置 z ≤ c \text{z}_{\leq c} z≤c,只预测 c c c后边的位置 z > c {\text{z}_{>c}} z>c。XLNet中切分点 c c c 的选择由超参数 K K K来确定, K ≈ n n − c K \approx \frac{n}{n-c} K≈n−cn,其中 n n n为序列长度。 K K K越大,则需要预测的token数量越少。
这就是Partial Prediction 部分预测,对于切分点 c c c之前的token无需计算query representation,这会大大节省内存,加快模型训练速度。加入切分点后,XLNet的目标函数将变为:
m a x θ E z ∼ Z l o g p θ ( x z > c ∣ x ≤ c ) = m a x θ E z ∼ Z [ ∑ t = c + 1 n l o g p θ ( x z t ∣ x z < t ) ] \begin{align} \mathop{max}_{\theta} \quad \mathbb{E}_{\text{z}∼\mathbb{Z}} log \, p_{\theta}(\text{x}_{z_{>c}}|\text{x}_{\leq c}) = \mathop{max}_{\theta}\quad \mathbb{E}_{\text{z}∼\mathbb{Z}} \left[ \sum_{t=c+1}^n log\;p_{\theta}(x_{z_t}|\text{x}_{\text{z}_{
许多下游任务存在多段输入的情况,比如QA任务中包含query( 简记为A )和answer (简记为B)两个部分,数据的输入形式同BERT一致: [CLS, A, SEP, B, SEP] \text{[CLS, A, SEP, B, SEP]} [CLS, A, SEP, B, SEP]。
但是在segment循环的时候,每个部分仅仅使用对应上下文的状态缓存。
Relative Segment Encoding (相对段编码) , 这里的Segment不是上边将的将序列划分为固定的Segment,而是指输入数据的不同部分,例如 [CLS, A, SEP, B, SEP] \text{[CLS, A, SEP, B, SEP]} [CLS, A, SEP, B, SEP], A \text{A} A和 B \text{B} B分别属于不同的Segment。
BERT直接使用了绝对编码,直接给 A \text{A} A和 B \text{B} B中的token依次设置了0和1,用来指示整个序列中 A \text{A} A和 B \text{B} B是不同的segment,即是不同的文本段,例如一个是query,另一个是answer。
XLNet与BERT不同,它使用了相对编码,给定序列中的两个位置 i i i和 j j j,判断这两个位置对应的token是否在同一个segment里面,如果两者在同一个segment里面, s i j = s + s_{ij}=s_+ sij=s+,否则 s i j = s − s_{ij}=s_- sij=s−。 当预测第 i i i个位置token的时候,需要计算用 i i i位置的向量向另一位置 j j j做attention获取分数,其按照如下公式计算:
α i , j = ( q i + b ) T s i j \alpha_{i,j} = (q_i+b)^T s_{ij} αi,j=(qi+b)Tsij
其中 q i q_i qi为第 i i i个位置的查询向量, b b b是一个可学习的参数。最终 a i , j a_{i,j} ai,j将被加到正常Self-Attention的注意力分数上。
使用相对段编码有这样的优势:
模型的泛化效果会更好;
在微调任务上,它支持超过两个segment输入的下游任务(虽然预训练过程中使用了两个segment);
相关资料
XLNet:Generalized Autoregressive Pretraining for Language Understanding
XLNet Github
BERT(Bidirectional Encoder Representation from Transformers)是2018年10月由Google AI研究院提出的一种预训练模型,该模型在机器阅读理解顶级水平测试SQuAD1.1中表现出惊人的成绩: 全部两个衡量指标上全面超越人类,并且在11种不同NLP测试中创出SOTA表现,包括将GLUE基准推高至80.4% (绝对改进7.6%),MultiNLI准确度达到86.7% (绝对改进5.6%),成为NLP发展史上的里程碑式的模型成就。
BERT的网络架构使用的是《Attention is all you need》中提出的多层Transformer结构,如 图1 所示。其最大的特点是抛弃了传统的RNN和CNN,通过Attention机制将任意位置的两个单词的距离转换成1,有效的解决了NLP中棘手的长期依赖问题。Transformer的结构在NLP领域中已经得到了广泛应用。
BERT整体框架包含pre-train和fine-tune两个阶段。pre-train阶段模型是在无标注的标签数据上进行训练,fine-tune阶段,BERT模型首先是被pre-train模型参数初始化,然后所有的参数会用下游的有标注的数据进行训练。
图1 BERT结构
BERT是用了Transformer的encoder侧的网络,encoder中的Self-attention机制在编码一个token的时候同时利用了其上下文的token,其中‘同时利用上下文’即为双向的体现,而并非想Bi-LSTM那样把句子倒序输入一遍。
在它之前是GPT,GPT使用的是Transformer的decoder侧的网络,GPT是一个单向语言模型的预训练过程,更适用于文本生成,通过前文去预测当前的字。
Embedding由三种Embedding求和而成:
Token Embeddings是词向量,第一个单词是CLS标志,可以用于之后的分类任务
Segment Embeddings用来区别两种句子,因为预训练不光做LM还要做以两个句子为输入的分类任务
Position Embeddings和之前文章中的Transformer不一样,不是三角函数而是学习出来的
其中[CLS]表示该特征用于分类模型,对非分类模型,该符号可以省去。[SEP]表示分句符号,用于断开输入语料中的两个句子。
BERT在第一句前会加一个[CLS]标志,最后一层该位对应向量可以作为整句话的语义表示,从而用于下游的分类任务等。因为与文本中已有的其它词相比,这个无明显语义信息的符号会更“公平”地融合文本中各个词的语义信息,从而更好的表示整句话的语义。 具体来说,self-attention是用文本中的其它词来增强目标词的语义表示,但是目标词本身的语义还是会占主要部分的,因此,经过BERT的12层(BERT-base为例),每次词的embedding融合了所有词的信息,可以去更好的表示自己的语义。而[CLS]位本身没有语义,经过12层,句子级别的向量,相比其他正常词,可以更好的表征句子语义。
BERT是用了Transformer的encoder侧的网络,如上图的transformer的Encoder部分,关于transformer的encoder的详细介绍可以参考链接:
在Transformer中,模型的输入会被转换成512维的向量,然后分为8个head,每个head的维度是64维,但是BERT的维度是768维度,然后分成12个head,每个head的维度是64维,这是一个微小的差别。Transformer中position Embedding是用的三角函数,BERT中也有一个Postion Embedding是随机初始化,然后从数据中学出来的。
BERT模型分为24层和12层两种,其差别就是使用transformer encoder的层数的差异,BERT-base使用的是12层的Transformer Encoder结构,BERT-Large使用的是24层的Transformer Encoder结构。
如上图将注意力看做不同的连线,它们用来连接被更新的位置(左半边)与被注意的位置(右半边)。不同的颜色分别代表相应的注意头,而线条颜色的深浅代表被注意的强度。
为了方便演示,这里采用以下例句:
句子A:I went to the store.句子B:At the store, I bought fresh strawberries.
BERT 用 WordPiece工具来进行分词,并插入特殊的分离符([CLS],用来分隔样本)和分隔符([SEP],用来分隔样本内的不同句子)。
因此实际输入序列为: [CLS] i went to the store . [SEP] at the store , i bought fresh straw ##berries . [SEP]
在这种模式中,每个位置主要注意序列中的下一个词(token)。下面将看到第2层0号头的一个例子。(所选头部由顶部颜色条中突出的显示色块表示。)
模式1:注意下一个词。
左:所有词的注意力。 右:所选词的注意力权重(“i”)
左边显示了所有词的注意力,而右侧图显示一个特定词(“i”)的注意力。在这个例子中,“i”几乎所有的注意力都集中在“went”上,即序列中的下一个词。
在左侧,可以看到 [SEP]符号不符合这种注意力模式,因为[SEP]的大多数注意力被引导到了[CLS]上,而不是下一个词。因此,这种模式似乎主要在每个句子内部出现。
该模式与后向RNN 有关,其状态的更新是从右向左依次进行。模式1出现在模型的多个层中,在某种意义上模拟了RNN 的循环更新。
在这种模式中,大部分注意力都集中在句子的前一个词上。例如,下图中“went”的大部分注意力都指向前一个词“i”。
这个模式不像上一个那样显著。有一些注意力也分散到其他词上了,特别是[SEP]符号。与模式1一样,这与RNN 有些类似,只是这种情况下更像前向RNN。
模式2:注意前一个词。
左:所有词的注意力。 右:所选词的注意力权重(“went”)
这种模式注意相同或相关的单词,包括其本身。在下面的例子中,第一次出现的“store”的大部分注意力都是针对自身和第二次出现的“store”。这种模式并不像其他一些模式那样显著,注意力会分散在许多不同的词上。
模式3:注意相同/相关的词。
左:所有词的注意力。 右:所选词的注意权重(“store”)
这种模式注意另一个句子中相同或相关的单词。例如,第二句中“store”的大部分注意力都指向第一句中的“store”。可以想象这对于下句预测任务(BERT预训练任务的一部分)特别有用,因为它有助于识别句子之间的关系。
模式4:注意其他句子中相同/相关的单词。
左:所有词的注意力。 右:所选词的注意权重(“store”)
这种模式似乎是更注意能预测该词的词,而不包括该词本身。在下面的例子中,“straw”的大部分注意力都集中在“##berries”上(strawberries 草莓,因为WordPiece分开了),而“##berries”的大部分注意力也都集中在“straw”上。
模式5:注意能预测该单词的其他单词。
左:所有词的注意力。 右:所选词的注意力(“## berries”)
这个模式并不像其他模式那样显著。例如,词语的大部分注意力都集中在定界符([CLS])上,而这是下面讨论的模式6的特征。
在这种模式中,词语的大部分注意力都集中在分隔符[CLS]或 [SEP]上。在下面的示例中,大部分注意力都集中在两个 [SEP]符号上。这可能是模型将句子级状态传播到单个词语上的一种方式。
模式6:注意分隔符。 左:所有词的注意力。 右:所选词的注意权重(“store”)
BERT是一个多任务模型,它的预训练(Pre-training)任务是由两个自监督任务组成,即MLM和NSP,如 图2 所示。
图2 BERT 预训练过程示意图
MLM是指在训练的时候随即从输入语料上mask掉一些单词,然后通过的上下文预测该单词,该任务非常像我们在中学时期经常做的完形填空。正如传统的语言模型算法和RNN匹配那样,MLM的这个性质和Transformer的结构是非常匹配的。在BERT的实验中,15%的WordPiece Token会被随机Mask掉。在训练模型时,一个句子会被多次喂到模型中用于参数学习,但是Google并没有在每次都mask掉这些单词,而是在确定要Mask掉的单词之后,做以下处理。
80%的时候会直接替换为[Mask],将句子 “my dog is cute” 转换为句子 “my dog is [Mask]”。
10%的时候将其替换为其它任意单词,将单词 “cute” 替换成另一个随机词,例如 “apple”。将句子 “my dog is cute” 转换为句子 “my dog is apple”。
10%的时候会保留原始Token,例如保持句子为 “my dog is cute” 不变。
这么做的原因是如果句子中的某个Token 100%都会被mask掉,那么在fine-tuning的时候模型就会有一些没有见过的单词。加入随机Token的原因是因为Transformer要保持对每个输入token的分布式表征,否则模型就会记住这个[mask]是token ’cute‘。至于单词带来的负面影响,因为一个单词被随机替换掉的概率只有15%*10% =1.5%,这个负面影响其实是可以忽略不计的。 另外文章指出每次只预测15%的单词,因此模型收敛的比较慢。
优点
1)被随机选择15%的词当中以10%的概率用任意词替换去预测正确的词,相当于文本纠错任务,为BERT模型赋予了一定的文本纠错能力;
2)被随机选择15%的词当中以10%的概率保持不变,缓解了finetune时候与预训练时候输入不匹配的问题(预训练时候输入句子当中有mask,而finetune时候输入是完整无缺的句子,即为输入不匹配问题)。
缺点
注意
BERT预训练模型最多只能输入512个词,这是因为在BERT中,Token,Position,Segment Embeddings 都是通过学习来得到的。在直接使用Google 的BERT预训练模型时,输入最多512个词(还要除掉[CLS]和[SEP]),最多两个句子合成一句。这之外的词和句子会没有对应的embedding。
如果有足够的硬件资源自己重新训练BERT,可以更改 BERT config,设置更大max_position_embeddings 和 type_vocab_size值去满足自己的需求。
在海量的语料上训练完BERT之后,便可以将其应用到NLP的各个任务中了。 微调(Fine-Tuning)的任务包括:基于句子对的分类任务,基于单个句子的分类任务,问答任务,命名实体识别等。
基于句子对的分类任务:
MNLI:给定一个前提 (Premise) ,根据这个前提去推断假设 (Hypothesis) 与前提的关系。该任务的关系分为三种,蕴含关系 (Entailment)、矛盾关系 (Contradiction) 以及中立关系 (Neutral)。所以这个问题本质上是一个分类问题,我们需要做的是去发掘前提和假设这两个句子对之间的交互信息。
QQP:基于Quora,判断 Quora 上的两个问题句是否表示的是一样的意思。
QNLI:用于判断文本是否包含问题的答案,类似于我们做阅读理解定位问题所在的段落。
STS-B:预测两个句子的相似性,包括5个级别。
MRPC:也是判断两个句子是否是等价的。
RTE:类似于MNLI,但是只是对蕴含关系的二分类判断,而且数据集更小。
SWAG:从四个句子中选择为可能为前句下文的那个。
基于单个句子的分类任务
SST-2:电影评价的情感分析。
CoLA:句子语义判断,是否是可接受的(Acceptable)。
问答任务
命名实体识别
图3 BERT 用于不同的 NLP 任务
如上图所示,图中的Trm代表的是Transformer层,E代表的是Token Embedding,即每一个输入的单词映射成的向量,T代表的是模型输出的每个Token的特征向量表示。
BERT使用的是双向的Transformer,OpenAI GPT使用的是从左到右的Transformer。ELMo使用的是单独的从左到右和从右到左的LSTM拼接而成的特征。其中只有BERT在所有的层考虑了左右上下文。除此之外,BERT和OpenAI GPT是微调(fine-tuning)的方法,而ELMo是一个基于特征的方法。
从网络结构以及最后的实验效果来看,BERT 比 ELMo 效果好主要集中在以下几点原因:
LSTM 抽取特征的能力远弱于 Transformer
拼接方式双向融合的特征融合能力偏弱
BERT 的训练数据以及模型参数均多于 ELMo
优点
BERT 相较于原来的 RNN、LSTM 可以做到并发执行,同时提取词在句子中的关系特征,并且能在多个不同层次提取关系特征,进而更全面反映句子语义。
相较于 word2vec,其又能根据句子上下文获取词义,从而避免歧义出现。
缺点
模型参数太多,而且模型太大,少量数据训练时,容易过拟合。
BERT的NSP任务效果不明显,MLM存在和下游任务mismathch的情况。
BERT对生成式任务和长序列建模支持不好。