大模型时代还不理解自注意力?这篇文章教你从头写代码实现

AI人工智能7个月前发布 ash
534 0


自注意力

是 LLM 的一大核心组件。对大模型及相关应用开发者来说,理解

自注意力

非常重要。近日,Ahead of AI 杂志运营者、

机器学习

和 AI 研究者 Sebastian Raschka 发布了一篇文章,介绍并用代码从头实现了 LLM 中的

自注意力

、多头注意力、交叉注意力和因果注意力。

太长不看版这篇文章将介绍 Transformer 架构以及 GPT-4 和 Llama 等大型

语言模型

(LLM)中使用的

自注意力

机制。

自注意力

等相关机制是 LLM 的核心组件,因此如果想要理解 LLM,就需要理解它们。

不仅如此,这篇文章还会介绍如何使用 Python 和 PyTorch 从头开始编写它们的代码。在我看来,从头开始写算法、模型和技术的代码是一种非常棒的学习方式!

考虑到文章篇幅,我假设读者已经知道 LLM 并且已经对

注意力机制

有了基本了解。本文的目标和重点是通过 Python 和 PyTorch 编程过程来理解

注意力机制

的工作方式。


介绍

自注意力


自注意力

自在原始 Transformer 论文《Attention Is All You Need》中被提出以来,已经成为许多当前最佳的

深度学习

模型的一大基石,尤其是在

自然语言处理

(NLP)领域。由于

自注意力

已经无处不在,因此理解它是很重要的。

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



原始 Transformer 架构,来自论文 https://arxiv.org/abs/1706.03762

究其根源,

深度学习

中的「注意力(attention)」概念可以追溯到一种用于帮助循环

神经网络

(RNN)处理更长序列或句子的技术。举个例子,假如我们需要将一个句子从一种语言翻译到另一种语言。逐词翻译的操作方式通常不可行,因为这会忽略每种语言独有的复杂语法结构和习惯用语,从而导致出现不准确或无意义的翻译结果。

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



上图是不正确的逐词翻译,下图是正确的翻译结果

为了解决这个问题,研究者提出了

注意力机制

,让模型在每个时间步骤都能访问所有序列元素。其中的关键在于选择性,也就是确定在特定上下文中哪些词最重要。2017 年时,Transformer 架构引入了一种可以独立使用的

自注意力

机制,从而完全消除了对 RNN 的需求。

(由于本文的重点是

自注意力

的技术细节和代码实现,所以只会简单谈谈相关背景。)

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



来自论文《Attention is All You Need》的插图,展示了 making 这个词对其它词的依赖或关注程度,其中的颜色代表注意力

权重

的差异。

对于

自注意力

机制,我们可以这么看:通过纳入与输入上下文有关的信息来增强输入嵌入的信息内容。换句话说,

自注意力

机制让模型能够权衡输入序列中不同元素的重要性,并动态调整它们对输出的影响。这对语言处理任务来说尤其重要,因为在语言处理任务中,词的含义可能会根据句子或文档中的上下文而改变。

请注意,

自注意力

有很多变体。人们研究的一个重点是如何提高

自注意力

的效率。然而,大多数论文依然是实现《Attention Is All You Need》论文中提出的原始的缩放点积

注意力机制

(scaled-dot product attention mechanism),因为对于大多数训练大规模 Transformer 的公司来说,

自注意力

很少成为计算瓶颈。

因此,本文重点关注的也是原始的缩放点积

注意力机制

(称为

自注意力

),毕竟这是实践中最流行和使用范围最广泛的

注意力机制

。但是,如果你对其它类型的

注意力机制

感兴趣,可以参阅其它论文:

  1. Efficient Transformers: A Survey:https://arxiv.org/abs/2009.06732

  2. A Survey on Efficient Training of Transformers:https://arxiv.org/abs/2302.01107

  3. FlashAttention:https://arxiv.org/abs/2205.14135

  4. FlashAttention-v2:https://arxiv.org/abs/2307.08691


对输入句子进行嵌入操作

开始之前,我们先考虑以下输入句子:「Life is short, eat dessert first」。我们希望通过

自注意力

机制来处理它。类似于其它类型的用于处理文本的建模方法(比如使用循环

神经网络



卷积

神经网络


),我们首先需要创建一个句子嵌入(embedding)。

为了简单起见,这里我们的词典 dc 仅包含输入句子中出现的词。在真实世界应用中,我们会考虑训练数据集中的所有词(词典的典型大小在 30k 到 50k 条目之间)。

输入:

sentence = 'Life is short, eat dessert first'

dc = {s:i for i,s
       in enumerate(sorted(sentence.replace(',', '').split()))}

print(dc)

输出:

{'Life': 0, 'dessert': 1, 'eat': 2, 'first': 3, 'is': 4, 'short': 5}

接下来,我们使用这个词典为每个词分配一个整数索引:

输入:

import torch

sentence_int = torch.tensor(
    [dc[s] for s in sentence.replace(',', '').split()]
)
print(sentence_int)

输出:

tensor([0, 4, 5, 2, 1, 3])

现在,使用输入句子的整数向量表征,我们可以使用一个嵌入层来将输入编码成一个实数向量嵌入。这里,我们将使用一个微型的 3 维嵌入,这样一来每个输入词都可表示成一个 3 维向量。

请注意,嵌入的大小范围通常是从数百到数千维度。举个例子,Llama 2 的嵌入大小为 4096。这里之所以使用 3 维嵌入,是为了方便演示。这让我们可以方便地检视各个向量的细节。

由于这个句子包含 6 个词,因此最后会得到 6×3 维的嵌入:

输入:

vocab_size = 50_000

torch.manual_seed(123)
embed = torch.nn.Embedding(vocab_size, 3)
embedded_sentence = embed(sentence_int).detach()

print(embedded_sentence)
print(embedded_sentence.shape)

输出:

tensor([[ 0.3374, -0.1778, -0.3035],
        [ 0.1794,  1.8951,  0.4954],
        [ 0.2692, -0.0770, -1.0205],
        [-0.2196, -0.3792,  0.7671],
        [-0.5880,  0.3486,  0.6603],
        [-1.1925,  0.6984, -1.4097]])
torch.Size([6, 3])


定义

权重

矩阵

现在开始讨论广被使用的

自注意力

机制,也称为缩放点积注意,这是 Transformer 架构不可或缺的组成部分。


自注意力

使用了三个

权重

矩阵,分别记为 W_q、W_k 和 W_v;它们作为模型

参数

,会在训练过程中不断调整。这些矩阵的作用是将输入分别投射成序列的

查询

、键和值分量。

相应的

查询

、键和值序列可通过

权重

矩阵 W 和嵌入的输入 x 之间的矩阵乘法来获得:


  • 查询

    序列:对于属于序列 1……T 的 i,有 q⁽ⁱ⁾=x⁽ⁱ⁾W_q

  • 键序列:对于属于序列 1……T 的 i,有 k⁽ⁱ⁾=x⁽ⁱ⁾W_k

  • 值序列:对于属于序列 1……T 的 i,有 v⁽ⁱ⁾=x⁽ⁱ⁾W_v

  • 索引 i 是指输入序列中的 token 索引位置,其长度为 T。

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



通过输入 x 和

权重

W 计算

查询

、键和值向量

这里,q⁽ⁱ⁾ 和 k⁽ⁱ⁾ 都是维度为 d_k 的向量。投射矩阵 W_q 和 W_k 的形状为 d × d_k,而 W_v 的形状是 d × d_v。

(需要注意,d 表示每个词向量 x 的大小。)

由于我们要计算

查询

和键向量的点积,因此这两个向量的元素数量必须相同(d_q=d_k)。很多 LLM 也会使用同样大小的值向量,也即 d_q=d_k=d_v。但是,值向量 v⁽ⁱ⁾ 的元素数量可以是任意值,其决定了所得上下文向量的大小。

在接下来的代码中,我们将设定 d_q=d_k=2,而 d_v=4。投射矩阵的初始化如下:

输入:

torch.manual_seed(123)

d = embedded_sentence.shape[1]

d_q, d_k, d_v = 2, 2, 4

W_query = torch.nn.Parameter(torch.rand(d, d_q))
W_key = torch.nn.Parameter(torch.rand(d, d_k))
W_value = torch.nn.Parameter(torch.rand(d, d_v))

(类似于之前提到的

词嵌入

,实际应用中的维度 d_q、d_k、d_v 都大得多,这里使用小数值是为了方便演示。)


计算非归一化的注意力

权重

现在假设我们想为第二个输入元素计算注意力向量 —— 也就是让第二个输入元素作为这里的

查询

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



对于接下来的章节,我们将重点关注第二个输入 x⁽²⁾。

写成代码就是这样:

输入:

x_2 = embedded_sentence[1]
query_2 = x_2 @ W_query
key_2 = x_2 @ W_key
value_2 = x_2 @ W_value

print(query_2.shape)
print(key_2.shape)
print(value_2.shape)

输出:

torch.Size([2])
torch.Size([2])
torch.Size([4])

然后我们可以推而广之,为所有输入计算剩余的键和值元素,因为下一步计算非归一化注意力

权重

时会用到它们:

输入:

keys = embedded_sentence @ W_key
values = embedded_sentence @ W_value

print("keys.shape:", keys.shape)
print("values.shape:", values.shape)

输出:

keys.shape: torch.Size([6, 2])
values.shape: torch.Size([6, 4])

现在我们已经拥有了所有必需的键和值,可以继续下一步了,也就是计算非归一化注意力

权重

ω,如下图所示:

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



计算非归一化的注意力

权重

ω

如上图所示,ω(i,j) 是

查询

和键序列之间的点积 ω(i,j) = q⁽ⁱ⁾ k⁽ʲ⁾。

举个例子,我们能以如下方式计算

查询

与第 5 个输入元素(索引位置为 4)之间的非归一化注意力矩阵:

输入:

omega_24 = query_2.dot(keys[4])
print(omega_24)

(注意,ω 是希腊字幕,在代码中的变量名是 omega。)

输出:

tensor(1.2903)

由于我们后面需要这些非归一化注意力

权重

ω 来计算实际的注意力

权重

,因此这里就以上图所示的方式为所有输入 token 计算 ω 值。

输入:

omega_2 = query_2 @ keys.T
print(omega_2)

输出:

tensor([-0.6004,  3.4707, -1.5023,  0.4991,  1.2903, -1.3374])


计算注意力

权重


自注意力

的下一步是将非归一化的注意力

权重

ω 归一化,从而得到归一化注意力

权重

α(alpha);这会用到 softmax 函数。此外,在通过 softmax 函数进行归一化之前,还要使用 1/√{d_k} 对 ω 进行缩放,如下所示:

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



计算归一化注意力

权重

α

按 d_k 进行缩放可确保

权重

向量的欧几里得长度都大致在同等尺度上。这有助于防止注意力

权重

变得太小或太大 —— 这可能导致数值不稳定或影响模型在训练期间

收敛

的能力

我们可以这样用代码实现注意力

权重

的计算:

输入:

import torch.nn.functional as F

attention_weights_2 = F.softmax(omega_2 / d_k**0.5, dim=0)
print(attention_weights_2)

输出:

tensor([0.0386, 0.6870, 0.0204, 0.0840, 0.1470, 0.0229])

最后一步是计算上下文向量 z⁽²⁾,即原始

查询

输入 x⁽²⁾ 经过注意力加权后的版本,其通过注意力

权重

将所有其它输入元素作为了上下文:

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



这个注意力

权重

特定于某一个输入元素,这里选择的是输入元素 x⁽²⁾。

代码是这样:

输入:

context_vector_2 = attention_weights_2 @ values

print(context_vector_2.shape)
print(context_vector_2)

输出:

torch.Size([4])
tensor([0.5313, 1.3607, 0.7891, 1.3110])

请注意,这个输出向量的维度(d_v=4)比输入向量(d=3)多,因为我们之前已经设定了 d_v > d。但是,d_v 的嵌入大小可以任意选择。



自注意力

现在,总结一下之前小节中

自注意力

机制的代码实现。我们可以将之前的代码总结成一个紧凑的 SelfAttention 类:

输入:

import torch.nn as nn

class SelfAttention(nn.Module):

    def __init__(self, d_in, d_out_kq, d_out_v):
        super().__init__()
        self.d_out_kq = d_out_kq
        self.W_query = nn.Parameter(torch.rand(d_in, d_out_kq))
        self.W_key   = nn.Parameter(torch.rand(d_in, d_out_kq))
        self.W_value = nn.Parameter(torch.rand(d_in, d_out_v))

    def forward(self, x):
        keys = x @ self.W_key
        queries = x @ self.W_query
        values = x @ self.W_value

        attn_scores = queries @ keys.T  # unnormalized attention weights
            attn_weights = torch.softmax(
            attn_scores / self.d_out_kq**0.5, dim=-1
        )

        context_vec = attn_weights @ values
        return context_vec

遵照 PyTorch 的惯例,上面的 SelfAttention 类会在 __init__ 方法中对

自注意力


参数

进行初始化,然后通过 forward 方法为所有输入计算注意力

权重

和上下文向量。我们可以这样使用这个类:

输入:

torch.manual_seed(123)

# reduce d_out_v from 4 to 1, because we have 4 heads
d_in, d_out_kq, d_out_v = 3, 2, 4

sa = SelfAttention(d_in, d_out_kq, d_out_v)
print(sa(embedded_sentence))

输出:

tensor([[-0.1564,  0.1028, -0.0763, -0.0764],
        [ 0.5313,  1.3607,  0.7891,  1.3110],
        [-0.3542, -0.1234, -0.2627, -0.3706],
        [ 0.0071,  0.3345,  0.0969,  0.1998],
        [ 0.1008,  0.4780,  0.2021,  0.3674],
        [-0.5296, -0.2799, -0.4107, -0.6006]], grad_fn=<MmBackward0>)

可以从第二行看到,其值与前一节中 context_vector_2 的值完全一样:tensor ([0.5313, 1.3607, 0.7891, 1.3110])。


多头注意力

如下图所示,可以看到 Transformer 使用了一种名为多头注意力的模块。

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



来自 Transformer 原始论文的多头注意力模块。

这种多头注意力与我们之前讨论的

自注意力

机制(缩放点积注意力)有何关联呢?

在缩放点积注意力中,要使用分别表示

查询

、键和值的三个矩阵来对输入序列执行变换。在讨论多头注意力时,这三个矩阵可被看作是单个注意力头。下图总结了之前讨论和实现过的单注意力头:

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



总结之前实现的

自注意力

机制。

顾名思义,多头注意力涉及到多个这样的头,每一个都由

查询

、键和值矩阵构成。这个概念类似于在

卷积

神经网络


中使用多个核,通过多个输出通道产生特征图。

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



多头注意力:有多个头的

自注意力



为了用代码呈现,我们可以为之前的 SelfAttention 类写一个 MultiHeadAttentionWrapper 类:

class MultiHeadAttentionWrapper(nn.Module):

    def __init__(self, d_in, d_out_kq, d_out_v, num_heads):
        super().__init__()
        self.heads = nn.ModuleList(
            [SelfAttention(d_in, d_out_kq, d_out_v)
              for _ in range(num_heads)]
        )

    def forward(self, x):
        return torch.cat([head(x) for head in self.heads], dim=-1)

d_*

参数

与 SelfAttention 类中的一样 —— 这里仅有的新输入

参数

是注意力头的数量:

  • d_in:输入特征向量的维度

  • d_out_kq:

    查询

    和键输出的维度

  • d_out_v:值输出的维度

  • num_heads:注意力头的数量

我们使用这些输入

参数

将 SelfAttention 类初始化 num_heads 次,并且使用一个 PyTorch nn.ModuleList 来存储这些 SelfAttention 实例。

然后,其前向通过过程涉及到将每个 SelfAttention 头(存储在 self.heads 中)独立地用于输入 x。然后,沿最后的维度(dim=-1)将每个头的结果连接起来。下面来看实际操作!

为了说明简单,首先我们假设有输出维度为 1 的单个 SelfAttention 头。

输入:

torch.manual_seed(123)

d_in, d_out_kq, d_out_v = 3, 2, 1

sa = SelfAttention(d_in, d_out_kq, d_out_v)
print(sa(embedded_sentence))

输出:

tensor([[-0.0185],
        [ 0.4003],
        [-0.1103],
        [ 0.0668],
        [ 0.1180],
        [-0.1827]], grad_fn=<MmBackward0>)

现在,我们将其扩展到 4 个注意力头:

输入:

torch.manual_seed(123)

block_size = embedded_sentence.shape[1]
mha = MultiHeadAttentionWrapper(
    d_in, d_out_kq, d_out_v, num_heads=4
)

context_vecs = mha(embedded_sentence)

print(context_vecs)
print("context_vecs.shape:", context_vecs.shape)

输出:

tensor([[-0.0185,  0.0170,  0.1999, -0.0860],
        [ 0.4003,  1.7137,  1.3981,  1.0497],
        [-0.1103, -0.1609,  0.0079, -0.2416],
        [ 0.0668,  0.3534,  0.2322,  0.1008],
        [ 0.1180,  0.6949,  0.3157,  0.2807],
        [-0.1827, -0.2060, -0.2393, -0.3167]], grad_fn=<CatBackward0>)
context_vecs.shape: torch.Size([6, 4])

从上面的输出可以看到,单

自注意力

头的输出就是多头注意力输出的

张量

的第一列。

请注意这个多头注意力得到的是一个 6×4 维的

张量

:我们有 6 个输入 token 和 4 个

自注意力

头,其中每个

自注意力

头返回一个 1 维输出。之前的

自注意力

一节也得到了一个 6×4 维的

张量

。这是因为我们将输出维度设为了 4,而不是 1。既然我们可以就在 SelfAttention 类中调整输出嵌入的大小,那么我们为什么在实践时需要多个注意力头?

增加单

自注意力

头的输出维度和使用多个注意力头的区别在于模型处理和学习数据的方式。尽管这两种方法都能提升模型表征数据的不同特征或不同方面的能力,但它们的方式却有根本性的差异。

例如,多头注意力中的每个注意力头都可以学习关注输入序列的不同部分,捕获数据中的不同方面或关系。这种表征的多样性是多头注意力成功的关键。

多头注意力的效率也能更高,尤其是使用并行计算时。每个头都可以独立处理,这使得它们非常适合 GPU 或 TPU 等擅长并行处理的现代硬件加速器。

简而言之,使用多个注意力头不仅可以提高模型的能力,还可以增强其学习数据中各种特征和关系的能力。举个例子,7B 的 Llama 2 模型使用了 32 个注意力头。


交叉注意力

在上面编写的代码中,我们设定了 d_q = d_k = 2 和 d_v = 4。也就是说,

查询

和键序列使用了同样的维度。尽管值矩阵 W_v 的维度往往与

查询

和键矩阵一样(正如 PyTorch 中的 MultiHeadAttention 类),但值维度可以选取任意数值。

由于维度有时候是很难记的,所以这里我们总结一下之前的内容。如下图所示,其中总结了单个注意力头的各种

张量

大小。

大模型时代还不理解自注意力?这篇文章教你从头写代码实现



另一个角度看之前实现的

自注意力

机制,这里关注的重点是矩阵维度

上图对应于 Transformer 中使用的

自注意力

机制。对于这种

注意力机制

,还有一点尚未讨论:交叉注意力。

大模型时代还不理解自注意力?这篇文章教你从头写代码实现

交叉注意力是什么,又与

自注意力

有何不同?


自注意力

处理的是同一个输入序列。交叉注意力则会混合或组合两个不同的输入序列。对于上面的原始 Transformer 架构,也就是左侧由编码器模块返回的序列和右侧由解码器部分处理过的输入序列。

注意,在使用交叉注意力时,两个输入序列 x_1 和 x_2 的元素数量可以不同。但是,它们的嵌入维度必须一样。

下图展示了交叉注意力的概念。如果我们设 x_1 = x_2,则其就等价于

自注意力

大模型时代还不理解自注意力?这篇文章教你从头写代码实现

(请注意,

查询

通常来自解码器,键和值通常来自编码器。)

怎么写它的代码呢?我们可以把之前的 SelfAttention 类的代码拿过来改一下:

输入:

class CrossAttention(nn.Module):

    def __init__(self, d_in, d_out_kq, d_out_v):
        super().__init__()
        self.d_out_kq = d_out_kq
        self.W_query = nn.Parameter(torch.rand(d_in, d_out_kq))
        self.W_key   = nn.Parameter(torch.rand(d_in, d_out_kq))
        self.W_value = nn.Parameter(torch.rand(d_in, d_out_v))

    def forward(self, x_1, x_2):           # x_2 is new
        queries_1 = x_1 @ self.W_query

        keys_2 = x_2 @ self.W_key          # new
        values_2 = x_2 @ self.W_value      # new

        attn_scores = queries_1 @ keys_2.T # new
         attn_weights = torch.softmax(
            attn_scores / self.d_out_kq**0.5, dim=-1)

        context_vec = attn_weights @ values_2
        return context_vec

CrossAttention 类和之前的 SelfAttention 类有如下区别:

  • forward 方法有两个不同输入:x_1 和 x_2。

    查询

    来自 x_1,而键和值来自 x_2。这意味着

    注意力机制

    在评估两个不同输入之间的互动。

  • 注意力分数的计算方式是计算

    查询

    (来自 x_1)和键(来自 x_2)的点积。

  • 类似于 SelfAttention,每个上下文向量都是值的加权和。然而,在 CrossAttention 中,这些值源自第二个输入(x_2),而

    权重

    基于 x_1 和 x_2 之间的交互。

来看看它的实际效果:

输入:

torch.manual_seed(123)

d_in, d_out_kq, d_out_v = 3, 2, 4

crossattn = CrossAttention(d_in, d_o
© 版权声明

相关文章

暂无评论

暂无评论...