当前位置:   article > 正文

常见注意力机制解析_se注意力机制

se注意力机制

1.Squeeze-and-Excitation(SE)

SE的主要思想是通过对输入特征进行压缩和激励,来提高模型的表现能力。具体来说,SE注意力机制包括两个步骤:Squeeze和Excitation。在Squeeze步骤中,通过全局平均池化操作将输入特征图压缩成一个向量,然后通过一个全连接层将其映射到一个较小的向量。在Excitation步骤中,使用一个sigmoid函数将这个向量中的每个元素压缩到0到1之间,并将其与原始输入特征图相乘,得到加权后的特征图。通过SE注意力机制,模型可以自适应地学习到每个通道的重要性,从而提高模型的表现能力。在实际应用中,SE注意力机制已经被广泛应用于各种深度学习模型中,取得了很好的效果。

 代码如下:

  1. import torch
  2. from torch import nn
  3. from torchstat import stat # 查看网络参数
  4. # 定义SE注意力机制的类
  5. class SE_block(nn.Module):
  6. def __init__(self, channel, ratio=16):
  7. super(SE_block, self).__init__()
  8. self.avg_pool = nn.AdaptiveAvgPool2d(1)
  9. self.fc = nn.Sequential(
  10. nn.Linear(channel, channel // ratio, bias=False),
  11. nn.ReLU(inplace=True),
  12. nn.Linear(channel // ratio, channel, bias=False),
  13. nn.Sigmoid()
  14. )
  15. def forward(self, x):
  16. b, c, _, _ = x.size()
  17. y = self.avg_pool(x).view(b, c)
  18. y = self.fc(y).view(b, c, 1, 1)
  19. return x * y
  20. if __name__ == '__main__':
  21. # 构造输入层
  22. inputs = torch.rand(2,320,32,32)
  23. # 获取输入通道数
  24. channel = inputs.shape[1]
  25. # 模型实例化
  26. model = SE_block(channel, ratio=16)
  27. # 前向传播查看输出结果
  28. outputs = model(inputs)
  29. print(outputs.shape) #[2, 320, 32, 32]
  30. # print(model) # 查看模型结构
  31. stat(model, input_size=[320,32,32]) # 查看参数,不需要指定batch维度

2.Convolutional Block Attention Module(CBAM)

CBAM用于自适应地调整关注图像中的关键区域。CBAM注意力机制由两个模块组成:通道注意力模块和空间注意力模块。通道注意力模块用于学习每个通道的重要性,使得网络更多地关注那些有意义的特征,同时消除那些无意义的特征。这个模块的输入是一个卷积层的输出,输出是经过权重归一化之后的特征图。具体地,对于每个通道,通道注意力模块使用全局平均池化操作来得到一个特征向量,然后通过两个全连接层学习该通道的重要性,并对最终的卷积层输出进行加权。

空间注意力模块用于学习图像的不同空间区域的重要性,即在不同位置上的注意力权值。这个模块的输入是卷积层的输出,输出是通过空间维度上的卷积操作和全连接操作,产生的每个空间位置的注意力矩阵。这个矩阵能够自适应地调整感受野大小并关注图像中的关键区域。

通道和空间注意力模块可以结合使用,构建CBAM注意力机制。通过这种注意力机制的应用,可以提高图像分类、目标检测和语义分割等任务的准确性。

代码如下:

  1. import torch
  2. from torch import nn
  3. from torchstat import stat # 查看网络参数
  4. # 定义CBAM注意力机制的类
  5. class ChannelAttention(nn.Module):
  6. def __init__(self, in_planes, ratio=8):
  7. super(ChannelAttention, self).__init__()
  8. self.avg_pool = nn.AdaptiveAvgPool2d(1)
  9. self.max_pool = nn.AdaptiveMaxPool2d(1)
  10. # 利用1x1卷积代替全连接
  11. self.fc1 = nn.Conv2d(in_planes, in_planes // ratio, 1, bias=False)
  12. self.relu1 = nn.ReLU()
  13. self.fc2 = nn.Conv2d(in_planes // ratio, in_planes, 1, bias=False)
  14. self.sigmoid = nn.Sigmoid()
  15. def forward(self, x):
  16. avg_out = self.fc2(self.relu1(self.fc1(self.avg_pool(x))))
  17. max_out = self.fc2(self.relu1(self.fc1(self.max_pool(x))))
  18. out = avg_out + max_out
  19. return self.sigmoid(out)
  20. class SpatialAttention(nn.Module):
  21. def __init__(self, kernel_size=7):
  22. super(SpatialAttention, self).__init__()
  23. assert kernel_size in (3, 7), 'kernel size must be 3 or 7'
  24. padding = 3 if kernel_size == 7 else 1
  25. self.conv1 = nn.Conv2d(2, 1, kernel_size, padding=padding, bias=False)
  26. self.sigmoid = nn.Sigmoid()
  27. def forward(self, x):
  28. avg_out = torch.mean(x, dim=1, keepdim=True)
  29. max_out, _ = torch.max(x, dim=1, keepdim=True)
  30. x = torch.cat([avg_out, max_out], dim=1)
  31. x = self.conv1(x)
  32. return self.sigmoid(x)
  33. class CBAM_block(nn.Module):
  34. def __init__(self, channel, ratio=8, kernel_size=7):
  35. super(CBAM_block, self).__init__()
  36. self.channelattention = ChannelAttention(channel, ratio=ratio)
  37. self.spatialattention = SpatialAttention(kernel_size=kernel_size)
  38. def forward(self, x):
  39. x = x * self.channelattention(x)
  40. x = x * self.spatialattention(x)
  41. return x
  42. if __name__ == '__main__':
  43. # 构造输入层
  44. inputs = torch.rand(2,320,32,32)
  45. # 获取输入通道数
  46. channel = inputs.shape[1]
  47. # 模型实例化
  48. model = CBAM_block(channel, ratio=16, kernel_size=7)
  49. # 前向传播查看输出结果
  50. outputs = model(inputs)
  51. print(outputs.shape) #[2, 320, 32, 32]
  52. # print(model) # 查看模型结构
  53. stat(model, input_size=[320,32,32]) # 查看参数,不需要指定batch维度

3.Efficient Channel Attention(ECA)

ECA是一种用于图像处理的注意力机制模型。它主要是通过对图像通道进行注意力调控,提高图像特征表示的有效性。具体来说,ECA注意力机制模型由两部分组成:全局平均池化和线性变换。全局平均池化可以对每个通道的信息进行汇聚,从而判断该通道中的信息是否关键;线性变换可以将通道的信息进行缩放和平移,使得关键信息得到更好的保留,非关键信息得到抑制。如果某个通道的信息对于图像表现并不关键,则可以对其进行抑制,以提高其他通道的表现。ECA注意力机制相比于其他注意力机制模型的优势在于其模型复杂度低、计算效率高、效果好。因此,它被广泛应用于图像分类、目标检测和图像分割等领域。

 代码如下:

  1. import torch
  2. import math
  3. from torch import nn
  4. from torchstat import stat # 查看网络参数
  5. # 定义ECA注意力机制的类
  6. class ECA_block(nn.Module):
  7. def __init__(self, channel, b=1, gamma=2):
  8. super(ECA_block, self).__init__()
  9. kernel_size = int(abs((math.log(channel, 2) + b) / gamma))
  10. kernel_size = kernel_size if kernel_size % 2 else kernel_size + 1
  11. self.avg_pool = nn.AdaptiveAvgPool2d(1)
  12. self.conv = nn.Conv1d(1, 1, kernel_size=kernel_size, padding=(kernel_size - 1) // 2, bias=False)
  13. self.sigmoid = nn.Sigmoid()
  14. def forward(self, x):
  15. y = self.avg_pool(x)
  16. y = self.conv(y.squeeze(-1).transpose(-1, -2)).transpose(-1, -2).unsqueeze(-1)
  17. y = self.sigmoid(y)
  18. return x * y.expand_as(x)
  19. if __name__ == '__main__':
  20. # 构造输入层
  21. inputs = torch.rand(2,320,32,32)
  22. # 获取输入通道数
  23. channel = inputs.shape[1]
  24. # 模型实例化
  25. model = ECA_block(channel)
  26. # 前向传播查看输出结果
  27. outputs = model(inputs)
  28. print(outputs.shape) #[2, 320, 32, 32]
  29. # print(model) # 查看模型结构
  30. stat(model, input_size=[320,32,32]) # 查看参数,不需要指定batch维度

4.Coordinate attention(CA)

CA可以避免全局pooling-2D操作造成的位置信息丢失,将注意力分别放在宽度和高度两个维度上,有效利用输入特征图的空间坐标信息。CA主要分为两个步骤:第一步是坐标信息的嵌入,给定输入X,使用池化层分别沿着水平坐标和垂直坐标对每个通道进行编码,得到一对方向感知特征图。第二步是坐标信息特征图的生成,首先将提取到的特征信息进行拼接,然后利用一个1×1卷积变换函数进行信息转换,进而得到中间特征图,并沿着空间维度分解为两个单独的张量,再利用两个卷积变换为具有相同通道数的张量,最后将输出结果进行扩展,分别作为注意力权重分配值。CA是一个简单灵活即插即用的模块,可以在不带来任何额外开销的前提下,提升网络的精度。

代码如下:

  1. import torch
  2. from torch import nn
  3. from torchstat import stat # 查看网络参数
  4. # 定义CA注意力机制的类
  5. class CA_Block(nn.Module):
  6. def __init__(self, channel, reduction=16):
  7. super(CA_Block, self).__init__()
  8. self.conv_1x1 = nn.Conv2d(in_channels=channel, out_channels=channel // reduction, kernel_size=1, stride=1,
  9. bias=False)
  10. self.relu = nn.ReLU()
  11. self.bn = nn.BatchNorm2d(channel // reduction)
  12. self.F_h = nn.Conv2d(in_channels=channel // reduction, out_channels=channel, kernel_size=1, stride=1,
  13. bias=False)
  14. self.F_w = nn.Conv2d(in_channels=channel // reduction, out_channels=channel, kernel_size=1, stride=1,
  15. bias=False)
  16. self.sigmoid_h = nn.Sigmoid()
  17. self.sigmoid_w = nn.Sigmoid()
  18. def forward(self, x):
  19. _, _, h, w = x.size()
  20. x_h = torch.mean(x, dim=3, keepdim=True).permute(0, 1, 3, 2)
  21. x_w = torch.mean(x, dim=2, keepdim=True)
  22. x_cat_conv_relu = self.relu(self.bn(self.conv_1x1(torch.cat((x_h, x_w), 3))))
  23. x_cat_conv_split_h, x_cat_conv_split_w = x_cat_conv_relu.split([h, w], 3)
  24. s_h = self.sigmoid_h(self.F_h(x_cat_conv_split_h.permute(0, 1, 3, 2)))
  25. s_w = self.sigmoid_w(self.F_w(x_cat_conv_split_w))
  26. out = x * s_h.expand_as(x) * s_w.expand_as(x)
  27. return out
  28. if __name__ == '__main__':
  29. # 构造输入层
  30. inputs = torch.rand(2,320,32,32)
  31. # 获取输入通道数
  32. channel = inputs.shape[1]
  33. # 模型实例化
  34. model = CA_Block(channel, reduction=16)
  35. # 前向传播查看输出结果
  36. outputs = model(inputs)
  37. print(outputs.shape) #[2, 320, 32, 32]
  38. # print(model) # 查看模型结构
  39. stat(model, input_size=[320,32,32]) # 查看参数,不需要指定batch维度

reference

http://t.csdn.cn/0XBy9icon-default.png?t=N3I4http://t.csdn.cn/0XBy9

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/290820
推荐阅读
相关标签
  

闽ICP备14008679号