简介

Convolutional Block Attention Module(CBAM), 卷积注意力模块。该论文发表在ECCV2018上(论文地址),这是一种用于前馈卷积神经网络的简单而有效的注意力模块。

CBAM融合了通道注意力(channel Attention)和空间注意力(Spatial Attention),同时该注意力模块非常轻量化,而且能够即插即用,可以用在现存的任何一个卷积神经网络中。

深度学习 注意力模块如何画 注意力模块作用_池化

 

CBAM的流程如上图所示

首先,输入是一个中间特征图,将特征图输入至Channel Attention Module 获取通道注意力,然后将注意力权重作用于中间特征图。

然后,将施加通道注意力的特征图输入至Spatial Attention Module 获取空间注意力,然后将注意力权重作用到特征图上。

最终,经过这两个注意力模块的串行操作,最初的特征图就经过了通道和空间两个注意力机制的处理,自适应细化特征。

深度学习 注意力模块如何画 注意力模块作用_池化_02

 

那么,CBAM的注意力到底如何计算如上图所示,我们将在下面进行讲解。

 

Channel Attention

深度学习 注意力模块如何画 注意力模块作用_池化_03

 

通道注意力,将特征图在空间维度池化,保留通道的特征信息。在CBAM中,我们可以看到,全局平均池化和全局最大值池化均有使用(SENet只使用了平均池化,而cbam认为最大值池化被可以捕捉突出特征,实验证明也确实是有效的)。

它的计算流程如下:

  1. 将输入的特征图分别进行全局最大池化和全局平均池化,将空间维度压缩为1,保留通道信息。
  2. 将两个池化后的特征送入共享的多层感知机(MLP)提取特征。
  3. 将经过MLP的池化特征相加,经过sigmoid激活得到最终的通道注意力权重。

 

Spatial Attention

深度学习 注意力模块如何画 注意力模块作用_池化_04

 

空间注意力,将特征图在通道维度池化,保留空间的特征信息。

它的计算流程如下:

  1. 将特征图(经过通道注意力计算后的)在通道维度分别进行最大值池化和平均池化,将通道维度压缩为1,保留空间信息。
  2. 将池化特征concatenate起来,经过一个卷积层提取特征,同时将通道维度降至1。
  3. 最终经过sigmoid激活,得到最终的空间注意力权重(包含进了通道注意力)。

 

在串行地进行完这两个步骤后,将空间注意力特征与原特征图相乘即可。从上述运算过程可以看出,CBAM作为独立的两个模块,可以直接添加在任何一个卷积神经网络中,带来的附加运算量开销也很小。

实现

接下来,我们用Pytorch实现CBAM。

Channel Attention

class ChannelAttention(nn.Module):
    def __init__(self, channels, reduction_radio=16):
        super().__init__()
        self.channels = channels
        self.inter_channels = self.channels  // reduction_radio
        self.maxpool = nn.AdaptiveMaxPool2d((1, 1))
        self.avgpool = nn.AdaptiveAvgPool2d((1, 1))

        self.mlp = nn.Sequential(  # 使用1x1卷积代替线性层,可以不用调整tensor的形状
            nn.Conv2d(self.channels, self.inter_channels,
                    kernel_size=1, stride=1, padding=0),
            nn.BatchNorm2d(self.inter_channels),
            nn.ReLU(),
            nn.Conv2d(self.inter_channels, self.channels,
                    kernel_size=1, stride=1, padding=0),
            nn.BatchNorm2d(self.channels)
        )
        self.sigmoid = nn.Sigmoid()

    def forward(self, x):  # (b, c, h, w)
        maxout = self.maxpool(x) # (b, c, 1, 1)
        avgout = self.avgpool(x) # (b, c, 1, 1)

        maxout = self.mlp(maxout) # (b, c, 1, 1)
        avgout = self.mlp(avgout) # (b, c, 1, 1)

        attention = self.sigmoid(maxout + avgout) #(b, c, 1, 1)

        return attention

 

Spatial Attention

class SpatialAttention(nn.Module):
    def __init__(self):
        super().__init__()
        self.conv = nn.Conv2d(in_channels=2, out_channels=1,
                kernel_size=7, stride=1, padding=3)
        self.sigmoid = nn.Sigmoid()
        
    def forward(self, x): # (b, c, h, w)
        maxpool = x.argmax(dim=1, keepdim=True) # (b, 1, h, w)
        avgpool = x.mean(dim=1, keepdim=True)   # (b, 1, h, w)

        out = torch.cat([maxpool, avgpool], dim=1) # (b, 2, h, w)
        out = self.conv(out)  # (b, 1, h, w)

        attention = self.sigmoid(out) #(b, 1, h, w)
        return attention

对于一个特征图X,CBAM的运算结果如下:

ca = ChannelAttention(64)
sa = SpatialAttention()

x = torch.randn(3, 64, 56, 56)

channel = ca(x)  # (3, 64, 1, 1)
x = channel * x  # (3, 64, 56, 56)

spatial = sa(x)  # (3, 1, 56, 56)
x = spatial * x  # (3, 64, 56, 56)

 

应用

在之前的一篇博客里,尝试使用Vision Transformer训练102种鲜花分类。但是因为从头训练Vision Transformer的效果不好,预训练权重又比较难获得,因此总体而言准确度较低。然后这次想用ResNet + CBAM重新试一下。最后也没太使劲调参,发现效果还不错。如何将CBAM插入到ResNet中如下图所示。

 

深度学习 注意力模块如何画 注意力模块作用_2d_05

 

同时想尝试一下,将CBAM的注意力权重提取出来可视化观察一下效果,那么接下来就写一下ResNet + CBAM的实现(以ResNet34为例)。

代码主要参考了这个仓库,不过因为要可视化等等,还是进行了一些修改。github

BasicBlock

class BasicBlock(nn.Module):
    expansion = 1  # 通道升降维倍数

    def __init__(self, in_channels, channels, stride=1, downsample=None, attention=None):
        super().__init__()

        self.conv1 = nn.Conv2d(in_channels, channels,
                               kernel_size=3, stride=stride, padding=1)  # 第一个卷积层,通过stride进行下采样
        self.bn1 = nn.BatchNorm2d(channels)
        self.conv2 = nn.Conv2d(channels, channels,
                               kernel_size=3, stride=1, padding=1)  # 第二个卷积层,不进行下采样
        self.bn2 = nn.BatchNorm2d(channels)

        self.downsample = downsample
        self.attention = attention  # CBAM模块
        self.stride = stride

        self.relu = nn.ReLU(inplace=True)

    def forward(self, x):
        residual = x

        out = self.bn1(self.conv1(x))
        out = self.relu(out)
        out = self.bn2(self.conv2(out))

        if self.attention is not None:
            out = self.attention[0](out) * out  # 先进行通道注意力
            self.attention_weights = self.attention[1](out)  # CBAM的注意力图
            out = self.attention_weights * out  # 然后进行空间注意力
        else:
            self.attention_weights = None

        if self.downsample is not None:
            residual = self.downsample(x)  # 通道数不变,1x1卷积层仅用于降采样

        out += residual
        return self.relu(out)

 

ResNet

class ResNet(nn.Module):
    def __init__(self, block, layers, num_classes=1000):
        self.in_channels = 64
        self.layers = layers
        super().__init__()

        self.conv1 = nn.Conv2d(3, 64, kernel_size=7, stride=2, padding=3, bias=False)
        self.bn1 = nn.BatchNorm2d(64)
        self.relu = nn.ReLU(inplace=True)
        self.maxpool = nn.MaxPool2d(kernel_size=3, stride=2, padding=1)

        self.layer1 = self._make_layer(block, 64, layers[0])  # 第一个残差层不进行下采样
        self.layer2 = self._make_layer(block, 128, layers[1], stride=2)
        self.layer3 = self._make_layer(block, 256, layers[2], stride=2)
        self.layer4 = self._make_layer(block, 512, layers[3], stride=2)

        self.attention_layer = [self.layer3, self.layer4]     # 仅在最后两个layer上添加注意力

        self.avgpool = nn.AdaptiveAvgPool2d((1, 1))
        self.flatten = nn.Flatten()
        self.fc = nn.Linear(512 * block.expansion, num_classes)

        for m in self.modules():
            if isinstance(m, nn.Conv2d):
                nn.init.xavier_normal_(m.weight, gain=1)
            elif isinstance(m, nn.BatchNorm2d):
                m.weight.data.fill_(1)
                m.bias.data.zero_()

    def _make_layer(self, block, channels, blocks, stride=1):  # block:basicblock or bottleneck
        downsample = None

        if stride != 1 or self.in_channels != channels * block.expansion:  # 需要下采样or要融合通道
            downsample = nn.Sequential(
                nn.Conv2d(self.in_channels, channels * block.expansion, kernel_size=1,
                          stride=stride, bias=False),
                nn.BatchNorm2d(channels * block.expansion)
            )
        layers = []
        layers.append(block(self.in_channels, channels, stride, downsample))  # 第一个残差块

        self.in_channels = channels * block.expansion
        for i in range(1, blocks):
            attention = None
            if i > 1: # 在第2层往后才添加cbam
                attention = nn.Sequential(
                    ChannelAttention(self.in_channels),
                    SpatialAttention())
              
            layers.append(block(self.in_channels, channels, attention=attention))

        return nn.Sequential(*layers)

    def forward(self, x):
        x = self.conv1(x)
        x = self.bn1(x)
        x = self.relu(x)
        x = self.maxpool(x)

        x = self.layer1(x)
        x = self.layer2(x)
        x = self.layer3(x)
        x = self.layer4(x)
        self._attention_weights = [None] * len(self.attention_layer)  # 将带有注意力层的注意力权重拿出来

        for i, layer in enumerate(self.attention_layer):
            for j, (name, blk) in enumerate(layer.named_children()):
                self._attention_weights[i] = blk.attention_weights  # 覆盖,仅获取最后一个block的注意力

        x = self.avgpool(x)
        x = self.flatten(x)
        x = self.fc(x)

        return x

    @property
    def attention_weights(self):
        return self._attention_weights

 

ResNet34 with CBAM

model_urls = {
    'resnet18': 'https://download.pytorch.org/models/resnet18-5c106cde.pth',
    'resnet34': 'https://download.pytorch.org/models/resnet34-333f7ec4.pth',
    'resnet50': 'https://download.pytorch.org/models/resnet50-19c8e357.pth',
    'resnet101': 'https://download.pytorch.org/models/resnet101-5d3b4d8f.pth',
    'resnet152': 'https://download.pytorch.org/models/resnet152-b121ed2d.pth',
}


def resnet34_cbam(pretrained=False, num_class=1000):

    model = ResNet(BasicBlock, [3, 4, 6, 3], num_class)
    if pretrained:
        pretrained_state_dict = model_zoo.load_url(model_urls['resnet34'])       # 预训练resnet的权重字典
        pretrained_state_dict = {k: v for k, v in pretrained_state_dict.items()  # 除去全连接层的预训练权重
                                 if (k in pretrained_state_dict and 'fc' not in k)}

        new_state_dict = model.state_dict()
        new_state_dict.update(pretrained_state_dict)  # 将预训练权重通过dict的update方式更新

        model.load_state_dict(new_state_dict)         # 将更新的网络权重载入到注意力resnet中

    return model

 

其他一些杂项

对于102种鲜花分类,只给了训练集和需要打标签的测试集。为了评估训练效果,最好从训练集中分出一部分来进行验证。本人就随机划分了10%的训练集用于验证,剩下的90%进行训练。

在训练的时候,从头开始训练相对来说容易过拟合。因此我们最好是迁移学习,使用自带的预训练权重,然后进行微调。然后把不带CBAM注意力模块的层的参数冻结(前面两层),让网络相对保持一个泛性(毕竟是在ImageNet上预训练的),不然相对而言还是比较容易过拟合。

在含有卷积注意力层的参数,使用较小的学习率进行微调。这样基本不用考虑太多其他的超参数,就可以得到一个很不错的效果(在这种方式下,第一次就得到了94.6%的准确率,比之前高了不少)。

 

整个训练代码以后应该会放在自己的github上,届时会在博客中贴出来。

 

可视化

因为添加了注意力,想尝试将注意力可视化出来观察一下效果。可视化的方法有很多,主要参考了这篇文章(知乎),将CBAM空间注意力的attention map(非特征图)可视化了出来,代码就不再放出来了。

在测试集中随机选取了一张图片,大体效果是这样的,感觉还凑合。红色感兴趣的部分竟然额能把几朵花的大部分都包含进去。


深度学习 注意力模块如何画 注意力模块作用_ide_06

                             

深度学习 注意力模块如何画 注意力模块作用_池化_07

 

可能以后再优化优化,用类别激活可视化(Class Activation Mapping,CAM)等方式将整个网络的特征图可视化一下。

 

若本文有错误的地方,欢迎大佬批评指正。