通常注意力机制可以使得网络资源集中到某几个需要关注的部分上,和有选择性的弱化对网络结果不重要的部分。网络的注意力机制来源于人的视觉注意力,因为人的精力有限,不能注意到所有的细节,而是有选择性的弱化不需要的或不重要的部分和集中到重要的部分上面,通过这种方式感知外部世界后,指挥人的眼睛移动[19]。注意力机制网络(Attention Mechanism)在一个网络任务中通过关注部分有用方式从而很大的降低了网络的复杂性。2014年,Mnih等人提出了视觉注意力机制的循环神经网络模型(Recurrent models of visual attention),发现加入注意力机制后的循环神经网络能够更好识别出图像中重要的部分,不仅使模型具有了高并发的网络计算能力而且还提高模型识别图像准确率[21]。2015年,Bahdanau D等人首次在自然语言处理领域加入了把注意力机制并且把它用到机器翻译和对齐任务,与传统神经网络模型相比,在准确率上有很大幅度的提升[22]。随后同年,Luong等人在Bahdanau的论文中的注意力机制基础上提出了两种注意力机制即全局注意力模型(Global attention model)和局部注意力模型(Local attention model),进一步提升了seq2seq这种自然语言处理任务的效果[23]。这两种注意力机制主要区别是全局注意力模型会利用query中每个token的向量值,而局部注意力模型仅仅利用其中某一部分token的向量值。2016年,Yin W等人提出基于卷机的注意力机制(attention based CNN, ABCNN),首次在卷积神经网络模型中引入了注意力机制对句子进行建模。在答案选择数据集(Answer Selection, AS)实验发现,分类的准确率有了很大的提升[24]。2017年,Vaswani等人通过单纯利用自注意力机制(Self-Attention)和全连接网络来训练语言翻译模型,与原有的神经网络模型相比,准确率更高[10]。
[21] Mnih V. Heess V. Graves A. Recurrent models of visual attention[C]//Proceedings of Advances in neural information processing systems 2014: 2204-2212
[24] Yin W , H Schütze, Xiang B , et al. ABCNN: Attention-Based Convolutional Neural Network for Modeling Sentence Pairs[J]. Computer Science, 2015.
[23] Luong M T, Pham H, Manning C D. Effective approaches to attention-based neural machine translation[C]//Proceedings of the 2015 conference on empirical methods in natural language processing. Lisbon, Portugal: Association for Computational Linguistics, 2015: 1412–1421.
[22] Bahdanau D , Cho K , Bengio Y . Neural Machine Translation by Jointly Learning to Align and Translate[J]. Computer Science, 2014.
[10] Vaswani A, Shazeer N, Parmar N, et al. Attention is all you need[C]//Advances in neural information processing systems. 2017: 5998-6008.
注意力机制(Attention Mechanism)是机器学习和人工智能领域中的一个重要概念,用于模拟人类视觉或听觉等感知过程中的关注机制。注意力机制的目标是让模型能够在处理信息时,更加关注与任务相关的部分,忽略与任务无关的信息。这种机制最初是受到人类大脑对信息处理的启发而提出的。
注意力机制的基本原理如下:
输入信息:首先,注意力机制接收输入信息,这可以是序列数据、图像、语音等。
查询、键和值:对于每个输入,注意力机制引入了三个部分:查询(query)、键(key)、值(value)。这些部分通常是通过神经网络学习得到的。查询用于表示要关注的内容,键用于表示输入信息中的特征,值则是与每个键相关的信息。
权重分配:注意力机制根据查询和键之间的关系来计算权重,这些权重决定了每个值在最终输出中的贡献程度。通常使用某种形式的相似度度量(如点积、缩放点积等)来计算权重。
加权求和:将计算得到的权重与对应的值相乘,然后将它们加权求和,得到最终的输出。这个输出通常包含了模型在处理输入信息时关注的部分。
重复:上述过程通常会被重复多次,以便模型可以在不同的上下文中动态地调整注意力。
注意力机制的核心思想是让模型能够自动地确定在处理输入信息时要关注哪些部分,从而提高了模型在各种任务中的性能。它在自然语言处理、计算机视觉和语音处理等领域都有广泛的应用,如在机器翻译中的Transformer模型、图像分割中的U-Net模型以及语音识别中的Listen, Attend and Spell(LAS)模型等。
总的来说,注意力机制可以帮助模型更好地理解和利用输入信息,提高了模型的表现和泛化能力。
由于LSTM和GRU只在一定程度上改进了循环神经网络的长句子依赖问题,并且信息的记忆能力也不是很强和计算能力有限。如果模型要记住很多信息,不得不设计的更复杂,为了解决这些问题,注意力机制出现了,它即能从大量信息中选择重要的信息来缓解神经网络模型的复杂度,而且能高效的并行运算。注意力机制的计算是一个匹配的过程,即通过一个查询(Query)向量到键(Key)和值(Value)对数据对来映射输出值.
注意力的计算一般有三个阶段。第一阶段是计算查询向量Q和每个输入的K的相关性或相似度,得到注意力权重系数 :
第二阶段是使用SoftMax函数对第一阶段得出的权重系数进行尺度缩放,即把它归一化为概率分布 ai ,分子是把神经元的当前输出映射到(0,+∞),分母是所有输出结果值的总和,公式如下:
第三阶段:将第二阶段得出的权重与value值加权求和,得到最终需要的Attention数值:
Attention
class
tf.keras.layers.Attention(use_scale=False, score_mode="dot", **kwargs)
Dot-product attention layer, a.k.a. Luong-style attention.
Inputs are query
tensor of shape [batch_size, Tq, dim]
, value
tensor of shape [batch_size, Tv, dim]
and key
tensor of shape [batch_size, Tv, dim]
. The calculation follows the steps:
[batch_size, Tq, Tv]
as a query
-key
dot product: scores = tf.matmul(query, key, transpose_b=True)
.[batch_size, Tq, Tv]
: distribution = tf.nn.softmax(scores)
.distribution
to create a linear combination of value
with shape [batch_size, Tq, dim]
: return tf.matmul(distribution, value)
.def softmax(t):
s_value = np.exp(t) / np.sum(np.exp(t), axis=-1, keepdims=True)
# print('softmax value: ', s_value)
return s_value
def numpy_attention(inputs,
mask=None,
training=None,
return_attention_scores=False,
use_causal_mask=False):
query = inputs[0]
value = inputs[1]
key = inputs[2] if len(inputs) > 2 else value
score = np.matmul(query, key.transpose())
attention_score_np = softmax(score)
result = np.matmul(attention_score_np, value)
print('attention score in numpy =', attention_score_np)
print('result in numpy = ', result)
def verify_logic_in_attention_with_query_value():
query_data = np.array(
[[1, 0.0, 1],[2, 3, 1]]
)
value_data = np.array(
[[2, 1.0, 1],[1, 4, 2 ]]
)
print(query_data.shape)
numpy_attention([query_data, value_data], return_attention_scores=True)
print("=============following is keras attention output================")
attention_layer= tf.keras.layers.Attention()
result, attention_scores = attention_layer([query_data, value_data], return_attention_scores=True)
print('attention_scores = ', attention_scores)
print('result=', result);
if __name__ == '__main__':
verify_logic_in_attention_with_query_value()
运行结果
(2, 3)
attention score in numpy = [[5.0000000e-01 5.0000000e-01]
[3.3535013e-04 9.9966465e-01]]
result in numpy = [[1.5 2.5 1.5 ]
[1.00033535 3.99899395 1.99966465]]
=============following is keras attention output================
attention_scores = tf.Tensor(
[[5.0000000e-01 5.0000000e-01]
[3.3535014e-04 9.9966466e-01]], shape=(2, 2), dtype=float32)
result= tf.Tensor(
[[1.5 2.5 1.5 ]
[1.0003353 3.998994 1.9996647]], shape=(2, 3), dtype=float32)
def verify_logic_in_attention_with_query_key_value():
query_data = np.array(
[[1, 0.0, 1],[2, 3, 1]]
)
value_data = np.array(
[[2, 1.0, 1],[1, 4, 2 ]]
)
key_data = np.array(
[[1, 2.0, 2], [3, 1, 0.1]]
)
print(query_data.shape)
numpy_attention([query_data, value_data, key_data], return_attention_scores=True)
print("=============following is keras attention output================")
attention_layer= tf.keras.layers.Attention()
result, attention_scores = attention_layer([query_data, value_data, key_data], return_attention_scores=True)
print(attention_layer.get_weights())
print('attention_scores = ', attention_scores)
print('result=', result);
if __name__ == '__main__':
verify_logic_in_attention_with_query_key_value()
结果
(2, 3)
attention score in numpy = [[0.47502081 0.52497919]
[0.7109495 0.2890505 ]]
result in numpy = [[1.47502081 2.57493756 1.52497919]
[1.7109495 1.86715149 1.2890505 ]]
=============following is keras attention output================
[]
attention_scores = tf.Tensor(
[[0.47502086 0.52497923]
[0.7109495 0.28905058]], shape=(2, 2), dtype=float32)
result= tf.Tensor(
[[1.4750209 2.5749378 1.5249794]
[1.7109495 1.8671517 1.2890506]], shape=(2, 3), dtype=float32)