CBAM(Convolutional Block Attention Module)

C B A M ( C o n v o l u t i o n a l B l o c k A t t e n t i o n M o d u l e ) CBAM(Convolutional Block Attention Module) CBAM(ConvolutionalBlockAttentionModule)

文章目录

C B A M CBAM CBAM注意力机制模块他本来的名字是卷积注意力机制模块 − C o n v o l u t i o n a l B l o c k A t t e n t i o n M o d u l e -Convolutional Block Attention Module −ConvolutionalBlockAttentionModule,该注意力机制是Sanghyun Woo 等人在2018 年在论文CBAM: Convolutional Block Attention Module提出来的。

CBAM论文地址

现在就对该算法进行一些解读和自己的理解,我将从原理,公式推导,以及代码结构进行阐述。

1. 背景原理

在很多的图像处理相关的领域卷积神经网络发挥了极为重要的作用,卷积神经网络以其强大的表征能力,极大的提高了视觉任务的性能。那么卷积神经网络的性能主要研究的三大因素是什么呢?什么是影响卷积神经网络性能的主要因素呢?

不难猜到就是下面三个:

  1. 深度 ( d e p t h ) (depth) (depth):卷积神经网络串联的总体层数

  2. 宽度 ( w i d t h ) (width) (width):卷积神经网络的通道个数

  3. 基数 ( c a r d i n a l i t y ) (cardinality) (cardinality):卷积神经网络的分支个数

对于深度这个方面,在之前 2016 2016 2016年由何凯明大神提出的 R e s N e t ResNet ResNet残差连接就已经实现了非常深的网络进行训练;

对于宽度这个方面,在之前 2015 2015 2015年由 g o o g l e google google公司提出的 G o o g l e N e t GoogleNet GoogleNet网络也已经实现。

对于基数这个方面,有更多的人都在进行努力。

但是除了这上面的几个因素之外,还引入了一个注意力的机制,可以让网络自动调节自己所重要的特征的权重。之前的 S E SE SE注意力机制主要关注的通道这方面的问题,但是我们是不是可以将通道和空间的注意力机制相互融合以实现一种新的注意力机制模块呢?

那么 C B A M CBAM CBAM模块油然而生。

这里就简单的讲一下什么是通道注意力,什么是空间注意力机制吧!

大家都知道在深度学习网络过程中,每个过程的图像批量实际上是四个维度的东西,分别是 ( B − B a t c h , C − c h a n n e l , H − h i g h t , W − w i d t h ) (B-Batch,C-channel,H-hight,W-width) (B−Batch,C−channel,H−hight,W−width),这四个维度就是每一次输入输出的玩意儿。

用大白话来说就是,一箱相册,一个箱子有很多图片册,而每个图片册有很多张图片,每个图片又有长度和宽度。之前我们主要关注的是图片册中每张图片整体的感觉;但是忽略了图片中的一些细节问题。

通道注意力主要就是在 − c h a n n e l -channel −channel也就是 C C C上做操作。

空间注意力主要就是在 − H -H −H和 W W W上面做操作。

  1. 通道注意力机制:关注是什么
  2. 空间注意力机制:关注在哪里

o k ok ok那么到现在我们就将通道空间 两个维度进行了很好的区分, C B A M CBAM CBAM的核心思想就是将这两个维度结合起来。

但是 C B A M CBAM CBAM并没有把它们一股脑混在一起处理,而是采用了 串联 串联 串联的方式。

这就好比我们找东西的逻辑:

  1. 先想找什么先过通道注意力模块
  2. 再看在哪里再过空间注意力模块

2.公式推导

这里主要有两组问题,也就是找什么东西和找的东西在哪里这两个问题。只有解决了这两个问题,才能够比较好的实现。

2.1.通道注意力

在这里要解决的是找什么东西的问题,也就是要解决的是那个通道更加的重要一些。

在 C B A M CBAM CBAM中,作者认为最大池化 ( M a x P o o l i n g ) (Max Pooling) (MaxPooling)和平均池化 ( A v e r a g e P o o l i n g ) (Average Pooling) (AveragePooling)包含了不同的信息特征,所以不同于 S E − N e t SE-Net SE−Net只用了平均池化, C B A M CBAM CBAM决定两个都要用。
F c = M c ( F ) = σ ( M L P ( A v g P o o l ( F ) ) + M L P ( M a x P o o l ( F ) ) ) = σ ( W 1 ( W 0 ( F a v g c ) + W 1 ( W 0 ( F m a x c ) ) F_c=M_c(F)=\sigma(MLP(AvgPool(F))+MLP(MaxPool(F)))=\sigma(W_1(W_0(F^{c}{avg})+W_1(W_0(F^{c}{max})) Fc=Mc(F)=σ(MLP(AvgPool(F))+MLP(MaxPool(F)))=σ(W1(W0(Favgc)+W1(W0(Fmaxc))

具体步骤:

  1. 对输入特征图 F F F分别进行全局平均池化全局最大池化 。这样就把 C × H × W C \times H \times W C×H×W的特征图变成了两个 C × 1 × 1 C \times 1 \times 1 C×1×1的向量。
    • F a v g c F_{avg}^c Favgc:平均池化得到的向量。
    • F m a x c F_{max}^{c} Fmaxc:最大池化得到的向量。
  2. 把经过池化后的向量分别放入到一组共享的全连接层 M L P MLP MLP。
  3. 经过 s i g m o i d sigmoid sigmoid函数归一化后和原始输入相乘。

2.2.空间注意力

解决了看什么,接下来要推导解决看哪里的空间注意力。

这时候输入的是已经经过通道注意力加权后的特征图 F ′ F' F′。
F s = M s ( F ′ ) = σ ( f 7 × 7 ( [ A v g P o o l ( F ) , M a x P o o l ( F ) ] ) ) = σ ( f 7 × 7 ( [ F a v g S , F m a x S ] ) ) F_s = M_s(F')=\sigma(f^{7\times 7}([AvgPool(F),MaxPool(F)]))=\sigma(f^{7\times 7}([F^{S}{avg},F^{S}{max}])) Fs=Ms(F′)=σ(f7×7([AvgPool(F),MaxPool(F)]))=σ(f7×7([FavgS,FmaxS]))

具体步骤:

  1. 压缩通道:这次我们要把通道维数压缩,所以在通道维度上分别做平均池化和最大池化。

    • 这就好比把厚厚的一叠相册(多个通道)压扁,看透视图。

    • 这就得到了两个二维的特征图: F a v g s F_{avg}^s Favgs 和 F m a x s F_{max}^s Fmaxs(大小都是 1 × H × W 1 \times H \times W 1×H×W)。

  2. 通道拼接 c o n c a t concat concat。

  3. 卷积提取: 使用一个 7 × 7 7 \times 7 7×7 的卷积核对拼接后的图进行卷积操作。

  4. 经过 s i g m o i d sigmoid sigmoid函数归一化后和原始输入相乘。

2.3.一些小问题

为什么通道注意力是使用直接相加的方式进行特征融合而空间注意力却是使用 c o n c a t concat concat叠加的方式进行特征融合?


3.CBAM结构框图和代码展示

3.1.结构框图

空间注意力模块 (关注 Where)
通道注意力模块 (关注 What)
2 x H x W
输入特征图 F

(H x W x C)
输入 F
全局平均池化
全局最大池化
共享 MLP
共享 MLP
相加 +
Sigmoid 激活
通道权重 Mc
乘 x
通道细化特征 F'
输入 F'
通道平均池化
通道最大池化
拼接 Concat
7x7 卷积
Sigmoid 激活
空间权重 Ms
乘 x
最终特征 F''
输出特征图

3.2.代码展示

python 复制代码
import torch 
import torch.nn as nn
import matplotlib.pyplot as plt
import torch.nn.functional as F
import torch.optim as optim
from torchvision import datasets, transforms
from torch.utils.data import DataLoader
import numpy as np

class CBAMBlock(nn.Module):
    '''CBAM注意力机制模块
        包含通道注意力机制模块以及空间注意力机制模块
        Args:
            outchannels: 输入特征图的通道数
            reduction: 通道注意力机制模块中全连接层的缩减比例
    '''
    def __init__(self , outchannels , reduction = 4):
        super(CBAMBlock , self).__init__()
        # 通道注意力机制模块 channel attention module
        self.maxpool = nn.AdaptiveMaxPool2d(1)
        self.avgpool = nn.AdaptiveAvgPool2d(1)
        self.fc1 = nn.Linear(outchannels , outchannels // reduction , bias = False)
        self.relu = nn.ReLU()
        self.fc2 = nn.Linear(outchannels // reduction , outchannels , bias = False)
        self.sigmoid = nn.Sigmoid()

        # 空间注意力机制模块 spatial attention module
        self.conv = nn.Conv2d(in_channels=2 , out_channels = 1 , kernel_size = 7 , stride = 1 , padding = 3 , bias = False)

    def forward(self , x):
        '''
            前向传播函数
            1. 通道注意力机制模块
                a. 分别进行最大池化以及平均池化
                b. 通过全连接层进行特征提取(分别)
                c. 将两个特征直接相加,然后sigmoid归一化
            2. 空间注意力机制模块
                a. 对通道注意力机制模块的输出进行最大池化以及平均池化
                b. 将两个池化结果在通道维度上进行拼接
                c. 通过卷积层进行特征提取,然后sigmoid归一化
        '''
        # 通道注意力机制模块
        maxpool_out = self.maxpool(x)
        avgpool_out = self.avgpool(x)

        maxpool_out = maxpool_out.view(-1 , x.size(1)) 
        avgpool_out = avgpool_out.view(-1 , x.size(1))

        F_max = self.fc2(self.relu(self.fc1(maxpool_out)))       
        F_avg = self.fc2(self.relu(self.fc1(avgpool_out)))

        channel_attention_out = self.sigmoid(F_max+F_avg).unsqueeze(2).unsqueeze(3)  # [B , C , 1 , 1] , add操作
        F_1 = channel_attention_out * x

        # 空间注意力机制模块
        spatial_maxpool , _ = torch.max(F_1 , dim=1 , keepdim = True)                       # [B , 1 , H , W]
        spatial_avgpool = torch.mean(F_1 , dim=1 , keepdim = True)                          # [B , 1 , H , W]
        spatial_attention_out = torch.cat([spatial_maxpool , spatial_avgpool] , dim = 1)    # [B , 2 , H , W] , concat操作
        spatial_attention_out = self.sigmoid(self.conv(spatial_attention_out))              # [B , 1 , H , W]
        F_2 = spatial_attention_out * F_1

        return F_2

Channel-wise Pooling
HxWx1
HxWx1
HxWx2
Channel-refined

Feature F'

(H x W x C)
MaxPool
AvgPool
Concatenate
Conv Layer

(7x7)
Sigmoid

σ
Spatial Attention Ms

(H x W x 1)

相关推荐
NAGNIP4 小时前
一文搞懂深度学习中的通用逼近定理!
人工智能·算法·面试
冬奇Lab5 小时前
一天一个开源项目(第36篇):EverMemOS - 跨 LLM 与平台的长时记忆 OS,让 Agent 会记忆更会推理
人工智能·开源·资讯
冬奇Lab5 小时前
OpenClaw 源码深度解析(一):Gateway——为什么需要一个"中枢"
人工智能·开源·源码阅读
AngelPP9 小时前
OpenClaw 架构深度解析:如何把 AI 助手搬到你的个人设备上
人工智能
宅小年9 小时前
Claude Code 换成了Kimi K2.5后,我再也回不去了
人工智能·ai编程·claude
九狼9 小时前
Flutter URL Scheme 跨平台跳转
人工智能·flutter·github
ZFSS9 小时前
Kimi Chat Completion API 申请及使用
前端·人工智能
天翼云开发者社区10 小时前
春节复工福利就位!天翼云息壤2500万Tokens免费送,全品类大模型一键畅玩!
人工智能·算力服务·息壤
知识浅谈11 小时前
教你如何用 Gemini 将课本图片一键转为精美 PPT
人工智能
Ray Liang11 小时前
被低估的量化版模型,小身材也能干大事
人工智能·ai·ai助手·mindx