yahei · 2020年02月12日

语义分割速览

原文链接:https://www.yuque.com/yahei/hey-yahei/segmentation


简单过一下语义分割的主流框架——FCN、UNet、SegNet、PSPNet、DeepLab

分割任务论文集与各方实现:https://github.com/mrgloom/awesome-semantic-segmentation
pytorch model zoo:https://github.com/yassouali/pytorch_segmentation
gluon model zoo:https://gluon-cv.mxnet.io/model_zoo/segmentation.html
SOTA Leaderboard:https://www.paperswithcode.com/task/semantic-segmentation

全卷积网络FCN

论文:《Fully Convolutional Networks for Semantic Segmentation(CVPR2015) 》
参考:《FCN的学习及理解 | CSDN, moonuke 》 
主要贡献:

  1. 在分类网络的基础上,取消要求固定输入长度的全连接FC,使网络能接受任意尺寸的输入
  2. 在网络深层部分使用反卷积层上采样,恢复深层特征的空间尺度
  3. 浅层特征注重细致的局部、位置信息,深层特征注重抽象的全局、分类信息。通过跨层连接,融合浅层(上采样前)和深层(上采样后)特征以提高网络的表现

为了跟图示统一,后续将以AlexNet 为backbone进行讨论。

从CNN到FCN

1581164890425-fffb7698-28a7-4706-984d-cf2884148ef7.png

就像《卷积神经网络CNN - 全连接层 | Hey~YaHei! 》所提到的,卷积神经网络CNN因为全连接层的限制,要求网络输入具有固定的尺寸大小。FCN作者将最后的三个全连接层换成1x1卷积,如果输入特征图恰好是1x1,那明显是等价的;如果不是1x1,那网络也不至于出错,但输出大小也会由输入大小决定。

反卷积上采样

FCN用反卷积(deconvolution)在网络深层做上采样操作,以恢复出输入图片同等尺寸的分割结果,也就是每个像素点的类别。
反卷积其实相当于零填充上采样+卷积,与padding不同的是,它填充在输入特征图的像素点之间。

3x3标准卷积3x3反卷积3x3空洞卷积
1581240688788-fecb5de8-9df2-4609-86d2-58d2fa19dd27.gif1581240688788-fecb5de8-9df2-4609-86d2-58d2fa19dd27.gif1581240688788-fecb5de8-9df2-4609-86d2-58d2fa19dd27.gif
(以上三图,蓝色方块为输入特征图的像素点,绿色方块为输出特征图像素点,空白部分填零)


关于反卷积的详细过程此处不再赘述,感兴趣可以参考《怎样通俗易懂地解释反卷积? | 知乎, 孙小也 》


值得一提的是,通常部署的时候不喜欢用反卷积,因为推理引擎往往没有针对反卷积做充分的优化。大多都直接用双线性插值/三线性插值做上采样(简化操作顺便提高推理速度),顶多再叠一层普通卷积来进一步提取特征。

跨层融合

众所周知,浅层特征注重细致的局部、位置信息,深层特征注重抽象的全局、分类信息。分类任务里不关注位置信息,所以随着网络前传,即使特征图分辨率越来越小,信息越来越抽象,位置信息逐步丢失,也无伤大雅。但检测任务和分割任务不同,除了需要给出对象的分类之外,还得给出位置信息——因此浅层特征的位置得想办法把它保留下来,比较直观的想法就是跨层把浅层和深层信息融合起来。

融合的方式有很多,最简单的如逐元素相加/相乘,或者连接特征(一般是从通道维度上做拼接)后做进一步的特征提取。FCN采用的即是简单的逐元素相加的形式,以500x500x3的输入图片为例(虚线以上就一个普通的全卷积网络,虚线以下是跨层融合相关的层)——
1581164890425-fffb7698-28a7-4706-984d-cf2884148ef7.png

  • 首先,由于不做分类任务,作为“分类器”的最后一层全连接()可以直接丢弃,保留前端的特征提取部分;
  • 分别将若干浅层特征图和最后的特征图抽离出来,经过卷积层特征映射到21通道的特征空间上(这里以VOC数据集为例,20正样本+1负样本(背景)=21分类),然后反卷积进行上采样(如图虚线以下的蓝色方块,有必要的话还得裁剪,通常是中央裁剪,到统一的尺寸),再对应逐元素相加(如图黄色方块)达到特征融合的目的;
  • 最后再一个反卷积上采样(配合裁剪)到原始尺寸,通道方向即相当于分类任务里的输出特征向量,决定着每个像素点所属的分类


训练细节

FCN采用普通的softmax交叉熵作为损失函数,既然通道方向决定了每个像素点的类别,那就对每个像素点计算softmax交叉熵,最后加和起来作为最终的损失进行训练。

原文倒是讲究,采用分阶段训练的方式,用与训练好的分类网络作为backbone,丢弃最后一层全连接,其他全连接替换成卷积并重新初始化权重(丢弃原有的全连接权重),再逐一融合中间层特征进行多阶段训练。

阶段训练部分
#11.png
#22.png
#33.png
#44.png

UNet

论文:《U-Net: Convolutional Networks for Biomedical Image Segmentation(MICCAI2015) 》
主要贡献:

  1. 以拼接+进一步提取特征的形式融合浅层(上采样前)与深层(上采样后)特征
  2. 用重叠切片的平铺策略对大尺度图像进行分割
  3. 采用加权的损失函数,加大边缘部分像素的权重,鼓励网络区分边界,有助于实例分割

拼接实现的跨层融合

1.png
网络整体结构呈现U型的对称结构,故称为UNet,左半部分为卷积组成的下采样路径,右半部分为反卷积和卷积组成的上采样路径。下采样路径每个阶段包含两_次_卷积和一次池化(卷积通道扩增+卷积等通道特征提取+池化下采样),上采样路径每个阶段包含一次反卷积和两次卷积(反卷积上采样并收缩通道+卷积通道收缩+卷积等通道特征提取)。

与FCN不同的是,UNet采取的是拼接(Concat)+进一步提取特征的特征融合方式,如上图上采样路径最左侧的蓝色外框空白填充的方块。相比粗暴的逐点相加,拼接更好的保留浅层的特征信息,但相对地也会增加计算的开销。

为了保证拼接的正确性,浅层特征图需要裁剪到目标尺寸,如上图下采样路径最右侧蓝色方块的虚线部分,这里通常采用的是中央裁剪。拼接后由于通道倍增,按照“特征图尺寸倍增,通道减半;尺寸减半,通道倍增”的惯例,需要先用卷积把通道数收缩到原来的一半。

重叠切片的平铺策略

UNet对大尺寸图像分割任务采用了重叠切片的平铺策略。

1.png

首先要注意到UNet与常规的网络不同,所有的卷积和池化都不加以padding(为了2x2池化下采样不加padding由不丢失信息,需要保证每次池化输入的尺寸为2的倍数),于是每做一次卷积,特征图都会稍微收缩一点点,这就导致了UNet的输出尺寸小于输入尺寸(如上图,572x572的输入最后出来只有392x392的掩膜)。

不padding意味着不引入无效信息,直观上是有好处的;另外这相当于用一张更大的图像来预测中央小区域的分割结果,相当于在分割的时候输入了目标区域以外的外围信息(如上图,蓝色框表示输入UNet的图像,最终只能产生黄色框部分的分割结果,实际在推理黄色框分割结果的时候也引用了黄色框以外蓝色框以内的外围信息的),这有助于提高模型的表现。

此外,UNet每步迭代采用单张图片输入,通过最大化输入图片尺寸来充分利用显存,同时对优化器采取一个较大的动量(如0.99)使之前的迭代结果能对本次迭代产生较大的影响,以此稳定训练过程。

加权损失惩罚边缘像素

2.png

首先看一下softmax交叉熵:

为了着重某些特殊的像素点,可以赋予一个权重,此时损失函数改造为



其中,
是权衡分类为每个分类所设置的一个损失权重;
也是人为设置的权重,如论文中推荐的
分别代表当前像素点到最近和次近的细胞的欧式距离
将计算出来的可视化后可以得到上图中的(d)

由于损失中对边缘位置的像素作出的较重的惩罚,最终将鼓励网络在实例边界做出较好的区分。


原文中还提到了对卷积参数采用标准差为的高斯分布初始化方式(如3x3卷积,输入通道数为64,有),这种方式其实跟He Initialization 也差不多。事实上自从BN层 的出现之后,深度学习网络对参数初始化也不再那么敏感。

变种和改进

UNet变种:《图像分割的U-Net系列方法 | 知乎, taigw
UNet++:《研习U-Net | 知乎, 周纵苇

SegNet

论文:《SegNet: A Deep Convolutional Encoder-Decoder Architecture for Image Segmentation(TPAMI2016) 》
参考:《SegNet图像分割网络直观详解 | 知乎, 郭冠华 》
主要贡献:用反池化替代反卷积进行上采样,简化上采样过程,降低计算开销

1.png
整体结构跟FCN和UNet其实差不多,主要差别在于上采样的手段变成了反池化。

反池化上采样

假设下采样路径采用的是最大池化,2x2池化操作如下图所示:
1581240688788-fecb5de8-9df2-4609-86d2-58d2fa19dd27.gif


每个滑窗只会采样最大值作为输出,反池化上采样则是反过来,把一个像素值填到一个2x2的输出框内,为了跟下采样对应,需要在做最大值池化的时候记录采样点的索引(如总体框架图上的Pooling Indices信息),反池化的时候则填到对应位置上的。其余三个像素点则直接填零,由后续的卷积层完成特征图的平滑处理。
1.png
2.png

反池化有三个优点:

  1. 改善了边缘轮廓的处理
  2. 减少参数数量和计算量
  3. 容易实现,可以很方便地应用到其他框架中

PSPNet

论文:《Pyramid Scene Parsing Network (CVPR2017) 》
参考:《论文笔记:Pyramid Scene Parsing Network | 简书, Efackw13 》
        《【图像分割模型】多感受野的金字塔结构—PSPNet | 知乎, 言有三 》
主要贡献:

  1. 从场景解析实际任务中发现了关系失配、相似分类混淆、不显眼物体难识别三个问题
  2. 在最后得到分割结果前,用混合尺度的池化获取不同感知区域的局部信息,加强网络对场景的感知

三个问题

作者在场景解析的实际任务当中发现FCN存在以下问题:

  1. 关系失配:比如在水面上识别出了汽车(如下图第一行所示)
  2. 相似分类混淆:比如把摩天大楼内部的一部分像素识别成普通建筑(如下图第二行所示)
  3. 不显眼物体难识别:比如在床被上识别不出相似花纹的枕头(如下图第三行所示)

1.png

金字塔池化模块(Pyramid Pooling Module)

为了解决上述三个问题,PSPNet提出了金字塔池化模块,该模块插入在输出分割结果前的最后一个特征图后边。

1.png

首先由CNN提取出特征图(PSPNet没像FCN, UNet, SegNet一样建立浅层到深层之间的跨层连接),然后经过不同的池化层下采样出不同尺寸的特征图(原文下采样为1x1, 2x2, 3x3, 6x6四种特征图),接着分别由卷积层将通道收缩为原来的1/N(原文中N=4)以保证拼接之后通道数与原来相同,由此得到不同尺寸的感知区域的局部信息。再将不同尺寸的局部信息上采样为原来特征图的尺寸但不改变通道数量(原文用双线性插值来上采样),与原始特征图拼接起来,最后经过卷积层映射到目标空间得到分割结果。

hszhao/PSPNet | github 为例,用netron可视化后可以看到详细的金字塔池化模块及后续处理的结构:
1.png

辅助损失

除了最终的分割分类损失之外,PSPNet还在中间位置加入了辅助损失,如下图所示,对ResNet第四阶段的输出特征图提前取出并且上采样到输入图片的尺寸,然后计算辅助损失loss2,并与主损失loss1加权求和后反传。
1.png
yassouali/pytorch_segmentation/trainer.py#L61 | github 设置了权重为0.4;
用于计算loss2的特征图的产生,具体也可以参见 yassouali/pytorch_segmentation/models/pspnet.py 的L65-L71L90-L94

DeepLab

参考:《【语义分割系列:一】DeepLab v1 / v2 论文阅读翻译笔记 | CSDN, 鹿鹿最可爱 》
        《【语义分割系列:五】DeepLab v3 / v3+ 论文阅读翻译笔记 | CSDN, 鹿鹿最可爱 》
        《deeplab系列总结(deeplab v1& v2 & v3 & v3+) | CSDN, Dlyldxwl 》

v1

论文:《Semantic Image Segmentation with Deep Convolutional Nets and Fully Connected CRFs (ICLR2015)
主要贡献:

  1. 结合CNN和PGM(概率图模型)
  2. 使用空洞卷积替代标准卷积,在保持计算量的同时增大感受野

(为了与原文对应,以下讨论均以VGG16为backbone)
_
1.png

两个问题

  1. 典型的CNN随着模型的深入需要逐渐下采样,这就导致信号(特征图)的分辨率逐渐减小;尽管可以通过减少下采样实现,但却带来另一个问题,即感受野偏小;
  2. 分类任务要求空间不变性,即图像经过空间变换(如旋转、平移)后识别的类别不发生改变;而分割任务不同,像素的定位信息也至关重要,如果图像经过空间变换,那么要求模型预测结果也要随之改变

减少下采样和使用空洞卷积

(缓解第一个问题)
按照常规利用分类网络做backbone的方式,作者剥离掉VGG16最后的三层全连接,此时最后一层卷积层的输出特征图分辨率为7x7,与输入的原始图像分辨率224x224相比,已经下采样了32倍,丢失了非常多的细节信息(原文中称之为很sparse、不dense)。

为了保留更多的信息,作者将最后两层池化的步长修改为1,也即取消了这两个池化的下采样功能,此时相当于只下采样了8倍。但如上一小节所说,这样就带来新的感受野偏小的问题。

DeepLab使用空洞卷积(dilated convolution)来解决这个问题,同时节约了不少的计算量。关于空洞卷积的图示和几种卷积的比较可以参考 FCN的反卷积上采样 一节。
1.png
(“下采样-标准卷积-上采样”和“空洞卷积”的效果比较,图源自v2)

2.png
(“标准卷积-下采样”和“空洞卷积”特征图尺寸变化示意,图源自以ResNet为backbone的v3)


简单来说,空洞卷积通过跳跃性地采样,能以3x3的卷积核得到5x5、7x7等更大的等效感受野。
前述两个取消下采样功能的池化层之后的卷积层就换成了成空洞卷积。

缓解了第一个问题后,论文同时指出随后的上采样不再需要反卷积来恢复分辨率,直接双线性插值就可以得到可观的结果。训练时直接对ground truth下采样8倍,然后与改造后的VGG16输出求交叉熵作为损失函数;预测时则直接双线性插值得到分割的结果。

全连接条件随机场后处理

(缓解第二个问题)
全连接条件随机场(Fully Connected Conditional Random Field, Fully Connected CRF)

1.png
(上下两行分别是softmax的输入和输出)

如上图的 DCNN output 一列所示,经过层层的下采样和上采样,特征图逐渐丢失部分信息,导致最终输出的图像显得比较平滑。而分割任务希望分割的结果边缘轮廓能够比较清楚犀利,于是作者引入了全连接CRF对CNN的输出结果进行后处理。

在传统的图像处理中,CRF通常用相邻像素来设计能量函数,从而消除一些噪音,达到平滑处理的目的。然而在分割中,我们的目标是恢复局部信息而非进一步平滑处理。
因此作者借鉴了《Efficient Inference in Fully Connected CRFs with Gaussian Edge Potentials (NIPS2011) 》全连接条件随机场来实现分割结果的锐化处理。其能量函数为

其中

对于一元项来说,是CNN产生的像素点的概率分布;
对于二元项来说,
,这意味着每个像素点都会和全图像的所有像素点建立联系,也即“全连接”;
求和项里是加权的应用在像素特征上的高斯核函数,原文采用像素点值和位置构造核函数

这里包含两项,第一项包含位置信息和值信息,后一项值考虑位置信息,两者通过加权;
也是人工设置的超参数

融合多尺度信息进行预测

和FCN、UNet一样,作者也尝试从浅层抽取特征与深层融合完成最后的分割预测。
具体来说,在输入图片和中间池化结果(前四个池化层的输出)上分别加两层卷积(3x3+1x1)做特征提取和通道缩放映射,使得分割效果得到少量的提升,但这个提升明显不如全连接CRF(两者可以兼容使用)。

v2

论文:《DeepLab: Semantic Image Segmentation with Deep Convolutional Nets, Atrous Convolution, and Fully Connected CRFs (CVPR2016)
主要贡献:对同一张特征图使用不同dilation的空洞卷积,并用多孔空间金字塔池化下采样到同一尺寸,以此融合多种感受野的特征信息。

多孔空间金字塔池化(Atrous Spatial Pyramid Pooling, ASPP)

思路很简单,其实是何恺明的空间金字塔池化的一个演化。关于空间金字塔池化(SPP)的内容可以回顾《漫谈池化层 - 空间金字塔池化 | Hey~YaHei! 》,此处不再赘述。
1.png
(图中rate指的就是我们常说的dilation,也即对输入特征图两个采样点的间隔,如标准卷积dilation=1)

v3

论文:《Rethinking Atrous Convolution for Semantic Image Segmentation (CVPR2017) 》
主要贡献:

  1. 归纳了四种常见的语义分割框架
  2. 改进了v2提出的ASSP

(原文以ResNet为backbone)
2.png

常见语义分割框架的比较

1.png

  1. 图像金字塔
    多尺度输入,小尺度输入响应语义,大尺度输入响应细节,最后融合多个结果,多个模型之间可以共享部分底层特征;显著缺点是模型冗余而庞大,推理慢开销大,训练麻烦
  2. 编码器-解码器
    深层捕获更加抽象的分类信息,辅之融合浅层特征恢复目标的细节尤其是空间信息
  3. 级联空洞卷积
    减少将采样保持特征图上的细节尤其是空间信息,利用空洞卷积扩大感受野
  4. 空间金字塔池化
    用不同dilation的并行空洞卷积提取不同感受野下的特征,最后用ASSP下采样到统一尺度进行特征融合

改进ASSP

2.png

  • 去掉一个“dilation=24的3x3卷积”分支
  • 增加一个“1x1的标准卷积”分支
  • 增加一个“全局平均池化 + 1x1标准卷积 + 双线性插值上采样”分支

v3+

论文:《Encoder-Decoder with Atrous Separable Convolution for Semantic Image Segmentation (ECCV2018)
主要贡献:

  1. 借鉴编码器-解码器架构,额外融合中间特征提高最终的分割能力
  2. 引入深度可分离卷积

(原文以使用了深度可分离卷积的Xception为backbone)
3.png

额外融合中间特征

1.png
如图,作者发现经过空间金字塔池化之后直接做一次八倍上采样过于粗暴,于是借鉴了编码器-解码器架构的思路,在空间金字塔池化后的特征图上采样四倍之后,与从中间层抽离出特征相融合,再做一次四倍上采样(注意编码器部分比原始模型要多一层,所以与原输入相比编码器的输出实际上下采样了十六倍而非八倍,换句话v3+加深了网络)

深度可分离卷积

可参考《漫谈卷积层 - 高效卷积 | Hey~YaHei!》,此处不再赘述。

损失函数

参考:《图像分割领域常见的loss fuction有哪一些? | 知乎, 小锋子Shawn 》

图像增广

工具:mdbloice/Augmentor 支持在变换原图的时候同步操作ground truth,很方便

推荐阅读
关注数
286
内容数
26
计算机视觉相关学习笔记,欢迎关注。[链接]
目录
极术微信服务号
关注极术微信号
实时接收点赞提醒和评论通知
安谋科技学堂公众号
关注安谋科技学堂
实时获取安谋科技及 Arm 教学资源
安谋科技招聘公众号
关注安谋科技招聘
实时获取安谋科技中国职位信息