FCOS长文详解

news2024/11/29 20:39:59

1. 概述

FCOS是一种one-stage、全卷积(Fully Convolutional)结构的目标检测模型,发表于2019年ICCV。(什么是one-stage?)
论文原地址:https://arxiv.org/abs/1904.01355
作者源码:https://github.com/tianzhi0549/FCOS
作者的源码有些复杂,我找了一个简单的版本https://github.com/zhenghao977/FCOS-PyTorch-37.2AP,作为本文的详解代码。

FCOS不同于在此之前热门的anchor based方法(比如R-CNN系列),没有设置anchor boxes来作为目标的候选区域,而是使用全卷积网络,结合FPN,直接拿去做检测,实现了anchor free,并达到了当时的state-of-art。当然anchor free这个概念不是作者提出来的,很早就有了。

作者在文中主要与anchor based检测方法作对比,首先指出anchor based方法的缺点:

  • 检测性能对anchor box的尺寸、高宽比、数量设置很敏感,而且这些都是超参数,而且人为调整;
  • 难以适应尺度变化大的目标,特别是小目标,因为anchor box限制了泛化能力,因此在新的任务中,需要重新设置anchor box的尺寸、高宽比等等;
  • 绝大多数anchor box都是负样本,这加剧了训练中正负样本不平衡的问题;
  • anchor boxes带来了巨大的计算量,比如计算IoU。

然后对比anchor based方法,阐述了FCOS的优点,其实就是把上边的缺点反过来就是了:

  • FCOS与许多全卷积网络可解决的问题在方法上具有一致性,方便在其他任务中复用;
  • anchor free,最大的贡献,减少了超参数,降低了训练难度;
  • 降低了计算量,速度快,因为没有anchor boxes;
  • FCOS也可以作为two-stage detectors使用,而且比anchor based的RPNs性能好。

2. 推理详细全流程

这一部先讲一下FCOS如何推理,每一步详细做了什么操作,后边再讲FCOS如何进行训练。

2.1 概括

文章作者给出了算法结构图,这也算是推理的流程图,非常重要的一张图

图1. 算法框架(图源:FCOS原文)

从图上明显看出来,FCOS包含了三个部分,分别是BackBone,Feature Pyramid, 还有一个Classification+Centerness+Regression。BackBone主要用于提取特征,作者使用的是ResNet;Feature Pyramid部分借鉴了FPN(feature pyramid networks, https://arxiv.org/abs/1612.03144),将BackBone提取出的三个不同尺寸的特征图进行融合,并输出五个不同尺寸的特征图像;最后将这五个特征图都拿去做识别回归,得到检测结果。

第三部分中的Classification和Regression好理解,分别是目标分类和bounding box 的回归,夹在中间的Centerness是作者的主要创新点之一,旨在减少低分目标框,具体的后边会说。

2.2 BackBone

代码里边的BackBone用的是ResNet50,这个东西我就不展开了,我们可以看一下代码上实现。

下图是ResNet网络的结构,其中,我把作者用到的输出层标注出来了,对应图1 -> Backbone的 C 3 C3 C3 C 4 C4 C4 C 5 C5 C5
在这里插入图片描述

图2. ResNet网络结构(图源:ResNet原文)

这是代码中的ResNet50实现:

class ResNet(nn.Module):
    def __init__(self, block, layers, num_classes=1000,if_include_top=False):
        self.inplanes = 64
        super(ResNet, self).__init__()
        self.conv1 = nn.Conv2d(3, 64, kernel_size=7, stride=2, padding=3,
                               bias=False)  #对应前图2上conv1
        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, 3)    #对应前图上conv2_x
        self.layer2 = self._make_layer(block, 128, 4, stride=2)    #对应前图2上conv3_x
        self.layer3 = self._make_layer(block, 256, 6, stride=2)    #对应前图2上conv4_x
        self.layer4 = self._make_layer(block, 512, 3, stride=2)    #对应前图2上conv5_x
        self.avgpool = nn.AvgPool2d(7, stride=1)
        if if_include_top:
            self.fc = nn.Linear(512 * block.expansion, num_classes)
        self.if_include_top=if_include_top

推理代码:

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

        x = self.layer1(x)
        C3 = self.layer2(x)
        C4 = self.layer3(C3)
        C5 = self.layer4(C4)
        return (C3,C4,C5)

这里提一下,论文里的输入图像 x x x尺寸是 [ 800 , 1024 ] [800, 1024] [800,1024] C 3 C3 C3 C 4 C4 C4 C 5 C5 C5的尺寸分别是 [ 100 , 128 ] [100, 128] [100,128], [ 50 , 64 ] [50, 64] [50,64], [ 25 , 32 ] [25, 32] [25,32],对于输入来说, C 3 C3 C3 C 4 C4 C4 C 5 C5 C5的stride分别是 8 8 8 16 16 16 32 32 32,这也是图1最左侧那些数字的含义。
在实际推理的时候,却不用严格按照 [ 800 , 1024 ] [800, 1024] [800,1024]来输入,因为是Backbone是全卷积结构,对输入图像的尺寸没有苛刻的硬性要求,比如输入尺寸 [ 885 , 1000 ] [885, 1000] [885,1000]也是ok的。

2.3 Feature Pyramid

Feature pyramid network是CVPR2017年的一篇文章,它在目标检测中融入了特征金字塔,提高了目标检测的准确率,尤其体现在小物体的检测上。FCOS作者使用FPN的原因是,FPN可以将不同尺度的目标分开检测,达到一个分而治之的效果。具体是怎么“分而治之”,这一节后边讲,先说一下FPN具体是什么东西,怎么实现的。
![在这里插入图片描述](https://img-blog.csdnimg.cn/f24f722260f74f1ea2797f05ce916f40.png#pic_center在这里插入图片描述

图3. FPN结构(图源:FPN原文)

图3 坐上部分对应的是图1的Backbone,是一个全卷积特征提取的过程,右边是一个自上而下的上采样+特征融合。

以图1来说, C 5 C5 C5经过一个 1 ∗ 1 1*1 11卷积得到 P 5 P5 P5 P 3 P3 P3经过 1 ∗ 1 1*1 11卷积,再与 P 5 P5 P5相加,得到 P 4 P4 P4,这里就有一个特征融合的过程了,由于 P 4 P4 P4 P 5 P5 P5的尺寸不一样,正好差了2倍,所以 P 5 P5 P5要经过2倍的上采样才能与 P 4 P4 P4相加,上采样使用的是最近邻采样;相加之后得到的 P 5 P5 P5还要经过一个 3 ∗ 3 3*3 33卷积层才能去做预测,就是图3红色的那部分。同理, P 3 P3 P3则是由 C 5 C5 C5 P 4 P4 P4融合而来。除此之外,FCOS作者做了一些改动,就是增加了 P 6 P6 P6 P 7 P7 P7,他们由 P 5 P5 P5经过 1 ∗ 1 1*1 11卷积层而来,stride为2。

上边,为什么要进行 1 ∗ 1 1*1 11卷积和 3 ∗ 3 3*3 33卷积呢?
我的理解是: 1 ∗ 1 1*1 11卷积核,最明显的目的是解决 C 3 C3 C3 C 4 C4 C4 C 5 C5 C5 通道数不同的问题,因为后边相加需要通道数相同才行,当然 1 ∗ 1 1*1 11卷积还有一些其他好处,我这里就不展开了,有兴趣的可以跳转这里; 3 ∗ 3 3*3 33卷积的作用,FPN原文是这么说的(下边的英文),意思是降低上采样带来的混叠效应。

we append a 3×3 convolution on each merged map to generate the final feature map, which is to reduce the aliasing effect of upsampling.

从FPN获取最终的特征图后,就要塞进Head获得最终结果了,FPN最后输出5个特征图,那么这5个特征图都要塞进Head网络,注意并不是5个不同的Head网络,而是同一个,网络结构、模型参数都是一样的。

好,到这里,FPN是怎么实现的说完了,我当时看到这里的时候,就有疑问,我拿啥去测?框呢,为啥没有候选框?
别急,下一小节慢慢道来

2.4 Classification+Centerness+Regression

如果是anchor based的算法,此时会以特征点为中心,划分出anchor boxes拿去回归;而FCOS直接对特征点对应原图的边框都进行回归。什么意思呢?就是先把特征点映射回原图像上,然后直接对这个点进行分类、回归。注意这里虽然是针对一个特征点进行预测,不代表FCOS是没有候选框的,或者说该特征点的感受野就是候选框。

映射关系为: ( x o r i g i n , y o r i g i n ) = ( x f e a t s + s 2 , y f e a t s + s 2 ) (x_{origin}, y_{origin})=(x_{feat}s+\frac{s}{2},y_{feat}s+\frac{s}{2}) (xorigin,yorigin)=(xfeats+2s,yfeats+2s)
其中, ( x f e a t , y f e a t ) (x_{feat},y_{feat}) (xfeat,yfeat)为特征点坐标, ( x o r i g i n , y o r i g i n ) (x_{origin}, y_{origin}) (xorigin,yorigin)映射回原图后的坐标, s s s为采样倍数stride

再举个例子,现在输入图像是640x640,这个特征图的stride是32,那么这个特征图的尺寸就是20x20,现在我们将特征点(3,4)映射回去,得到该特征点对应的候选框中心为(112,144),这个候选框的尺寸是32x32

此外,候选框的个数就是FPN输出的5个特征图上所有特征点数量,一个特征点对应一个候选框。

好了,说完候选框的由来,继续说由特征图生成最终结果。

Head网络也是全卷积网络,中间是什么结构我就不管了,主要关注模型的输出。输出结果分为三个部分,Classification、Centerness和Regression。
Classification:表示分类结果,尺寸与类别数量有关,如果是20类,就是每一类的得分;
Regression:表示目标位置偏移,相对于候选框,四个值分别表示相对于候选框的上下左右偏移;假设回归结果是 ( l , t , r , b ) (l,t,r,b) (l,t,r,b),候选框左上角坐标是 ( x 1 , y 1 ) (x1,y1) (x1,y1),右下角坐标是 ( x 2 , y 2 ) (x2,y2) (x2,y2),那么回归后的目标位置就是:左上: ( x 1 + l , y 1 + t ) (x1+l,y1+t) (x1+l,y1+t),右下: ( x 2 + r , y 2 + b ) (x2+r,y2+b) (x2+r,y2+b)
在这里插入图片描述

Centerness:表示Regression结果与候选框中心的偏离度,作者认为距离目标中心较远的位置产生很多低质量的预测边框,因此搞了个Centerness来剔除这部分低质量框。回归后的目标框中心相对于原来的中心肯定有偏移,除非 l = r , t = b l=r,t=b l=r,t=b,Centerness就是衡量这个偏离多远的一个指标,是一个0~1之间的数字,值越小表示偏离越大。在后处理中,它还要与Classification结果相乘,作为最终的分类置信度,这个值才拿去与设置的阈值比较。

3. 如何训练

FCOS训练的时候,首先要走一遍推理流程,然后根据推理结果和标签值计算损失函数,最后进行反向传播。

所以,这一节最主要的还是损失函数的计算,这个又牵扯到anchor free里边比较特殊的一部分:正负样本的划分,因此,这一部分主要讲解两个问题:正负样本如何划分LOSS函数

3.1 正负样本如何划分

如果你了解anchor based检测算法,比如yolov3/4/5,那你应该知道他们正负样本划分的依据是IOU,但是FCOS不是这么划分,因为它压根不计算IOU。FCOS的正负样本划分可以分为三步:
1、位置
再看一眼前边的映射公式,最朴素的思想就是:如果一个特征点对应的 ( x o r i g i n , y o r i g i n ) (x_{origin}, y_{origin}) (xorigin,yorigin)落在gt内,就认为该特征点为正样本。例如下图
在这里插入图片描述

黄色点都是正样本,他们都在gt里边,白色点均为负样本。

在源码中,发现了作者进行了改进,并不是所有在gt内的特征点都给正样本标签,而是以gt中心画个圆,半径是1.5倍的stride,在该圆内的特征点作为正样本,这样可以避免边缘区的低质量正样本。

无论画不画圆,这种方法都有一个问题,如果一个特征点同时落在两个gt中,如何分配标签呢?作者称这种现象为ambiguous point(如下图),我们针对这个问题进行改进
在这里插入图片描述
2、尺寸
FCOS非常巧妙地利用FPN层解决ambiguous point这个问题,FPN有两个作用:一、多尺度的特征融合,二、分而治之。FCOS就是利用了FPN的分而治之的作用,把多个尺度的目标分到FPN的多个输出特征层。

具体来说,FPN输出共5层特征层,分配到每层特征层上目标的长宽限制范围依次为:(-1, 64), (64, 128), (128, 256),(256, 512),(512, INF) ,对应的stride分别是[128,64,32,16,8]

作者在代码中是这么实现的,计算5个特征层各个特征点与该gt的 ( l , t , r , b ) (l,t,r,b) (l,t,r,b),保留 ( l , t , r , b ) (l,t,r,b) (l,t,r,b)四个值均大于0的点(特征点在gt内部);计算 ( l , t , r , b ) (l,t,r,b) (l,t,r,b)中的最大值,该最大值落在哪个区间,就由哪个特征层的特征点去预测。

如果还存在ambiguous point,俩gt尺寸相近且有重叠部分,就将小的那个gt的标签赋给这个特征点。这个地方乍一看处理的很粗糙,但是结合下边的Centerness就好很多。

3. Centerness

又讲到Centerness了,待会儿还要提到它,因为损失函数里边还有他一部分呢。

前边提到一次,作者认为距离目标中心较远的位置产生很多低质量的预测边框,即处于gt的边缘,提取的图像特征不正确,自然检测出低质量的框。在训练部分,FCOS会每个正样本赋予一个Centerness标签,用于表示该特征点的质量,计算方式为: c e n t e r n e s s ∗ = m i n ( l ∗ , r ∗ ) m a x ( l ∗ , r ∗ ) ∗ m i n ( t ∗ , b ∗ ) m a x ( t ∗ , b ∗ ) centerness^*=\sqrt{\frac{min(l^*,r^*)}{max(l^*,r^*)}*\frac{min(t^*,b^*)}{max(t^*,b^*)}} centerness=max(l,r)min(l,r)max(t,b)min(t,b) centerness越小,特征点离gt中心越远,质量越低。
在这里插入图片描述
虽然一个gt,可以会给好多个特征点赋予正样本的标签,但是根据Centerness,就能把这些正样本区别开来。

3.2 Loss函数

这里偷个懒,直接放论文里的公式了
在这里插入图片描述
说实话,没太看懂,别的博文也都只是贴出来,不给个讲解,好在我翻代码大致懂了loss函数咋算的

loss函数分为三部分,分类loss,位置loss和centerness loss。

  • 分类loss:采用focal loss,一种改进版的交叉熵,就是给不同样本赋予不同的权重,对于难学习的样本,权重大一点;
  • 位置loss:计算iou损失
  • centerness loss:标签值是公式计算出来的,跟网络输出的centerness计算交叉熵

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1668002.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

告别数据泥潭:PySpark性能调优的黄金法则

阿佑今天给大家带来个一张藏宝图——使用PySpark进行性能调优的黄金法则,从内存管理到执行计划,再到并行度设置,每一步都是提升数据处理速度的关键! 文章目录 Python Spark 详解1. 引言2. 背景介绍2.1 大数据处理技术演变2.2 Apac…

小猫咪邮件在线发送系统源码,支持添加附件

一款免登录发送邮件,支持发送附件,后台可添加邮箱,前台可选择发送邮箱 网站数据采取本地保存,所以使用前请给网站修改权限,否则很多功能将无法使用 安装教程: 1.上传服务器或者主机 2.登录后台,添加发送…

胆子真大,敢搞B站

今天给大家分享一款浏览器插件,能让你的B站在电脑端访问时候会更高级 作者已经开源到Github Star数量还在持续上升中 来看下这款插件究竟具备哪些功能 首先是开启首页干净模式,也就是去除大屏 正常情况我们访问B站是这个样子的~ 开启总开关后 首页的视…

【笔记】从零开始做一个男性人体的流程/躯干篇(超级详细)

躯干整体 大体 1.创建一个正方体,摆好位置 2.实例呀啥的都搞好 3.胸部它是一个前窄后宽的结构 斜方肌 臀部 1.臀部是前宽后窄的结构 2.我们再去侧面调整以下 胸椎向上倾斜,盆骨向下倾斜。脊椎是s形的 3.真实的身体没有这么方正,所以微调…

Adobe Premiere Pro安装

一、安装包下载 链接:https://pan.baidu.com/s/1aYqTSQQutDguKYZE-yNHiw?pwd72l8 提取码:72l8 二、安装步骤 1.鼠标右击【Pr2024(64bit)】压缩包(win11及以上系统需先点击“显示更多选项”)【解压到 Pr2024(64bit)】。 2.打开…

双向链表(双向带头循环)的增删查改的实现(简单易懂)

一:双向链表的概念 每个节点除开存有数据,还有一个指针指向前一个节点,一个指针指向后一个节点,尾节点和哨兵位互相指向,从而形成一个循环。 二:双向链表的实现第一点: 本文采用三个文件进行实…

Kexp 动态展示 k8s 资源对象依赖关系

kexp[1] 旨在以可视化的方式帮助用户理解和探索 Kubernetes 的能力。 适用场景: 学习和探索 Kubernetes 的功能。 应用开发,提供每个应用的对象图预设。 控制器和操作器的开发,支持动态对象图。 即将推出类似 Postman 的 Kubernetes API …

springboot实现文件防盗链设计

shigen坚持更新文章的博客写手,擅长Java、python、vue、shell等编程语言和各种应用程序、脚本的开发。记录成长,分享认知,留住感动。 个人IP:shigen 👋👋👋hello,伙伴们好久不见&…

《动手学深度学习》V2(11-18)

文章目录 十一、二 模型选择与过拟合和欠拟合1、模型的选择2、过拟合和欠拟合3、估计模型容量4、线性分类器的VC维5、过拟合欠拟合的代码实现 :fire:①生成数据集②定义评估损失③定义训练函数④三阶多项式函数拟合⑤线性函数拟合(欠拟合)⑤高阶多项式函数拟合(过拟合) 十三、权…

数据库脚本编写规范(SQL编写规范)

编写本文档的目的是保证在开发过程中产出高效、格式统一、易阅读、易维护的SQL代码。 1 编写目 2 SQL书写规范 3 SQL编写原则 软件开发全文档获取:点我获取

2024中国(重庆)VR/AR科技展8月举办

2024中国(重庆)VR/AR科技展8月举办 邀请函 主办单位: 中国航空学会 重庆市南岸区人民政府 招商执行单位: 重庆港华展览有限公司 展会背景: 2024中国航空科普大会暨第八届全国青少年无人机大赛在重庆举办,同时举办第二届中国…

FPGA SDRAM读写控制器

感谢邓堪文大佬 ! SDRAM 同步动态随机存取内存(synchronousdynamic randon-access menory,简称SDRAM)是有一个同步接口的动态随机存取内存(DRAM)。通常DRAM是有一个异步接口的,这样它可以随时响…

CSS之浮动

目录 浮动常见网页布局标准流(普通流、文档流)为什么需要浮动什么是浮动浮动特性(重难)注意:清除浮动 浮动 常见网页布局 本质:用CSS来摆放盒子,把盒子摆放到相应的位置 三种常见布局方式&…

Java常见数据结构---八大结构

前言: 数据结构是计算机底层存储、组织数据的方式。是指数据相互之间是以什么方式排列在一起的。 通常情况下,精心选择的数据结构可以带来更高的运行或者存储效率 常见的八大数据结构: 栈: 思想: 栈是一种数据结构&…

Python模块之Numpy(二)-- 生成各种随机数

对于 NumPy,与随机数相关的函数都在 random 模块中,其中包括可以生成服从多种概率分布随机数的函数,示例如下: #生成10个服从0-1均匀分布的随机数 arr1 np.random.random([2,5]) #也生成10个服从0-1均匀分布的随机数 arr2 np.ra…

TMS320F280049 CLB模块--LUT4 OUTLUT(4)

LUT4 示意图如下: OUTLUT 示意图如下: 寄存器 参考文档: TMS320F28004x Real-Time Microcontrollers Technical Reference Manual (Rev. G)

网络安全之OSI七层模型详解

OSI七层模型分为控制层(前三层)和数据层(后四层)。从第七层到一层为; 应用层(7)接收数据 表示层(6)将数据翻译为机器语言 会话层(5)建立虚连接…

http协议 tomcat如何访问资源 servlet理论介绍

tomcat介绍 bin是启动命令; conf是配置,可以修改端口号; lib是依赖的jar包; logs是日志 webapps是重点,在这里新建我们自己的javaWeb项目 tomcat如何访问资源 tomcat通过统一资源定位符(URL)来…

开源软件托管平台gogs操作注意事项

文章目录 一、基本说明二、gogs私有化部署三、设置仓库git链接自动生成参数四、关闭新用户注册入口 私有化部署gogs托管平台,即把gogs安装在我们自己的电脑或者云服务器上。 一、基本说明 系统环境:ubuntu 20.4docker安装 二、gogs私有化部署 前期准…

权限及权限操作

1.命令行解释器 Linux将命令行解释器称为外壳程序shell 命令行解释器的作用就是将用户输入的指令转换为操作系统能够直接执行的指令。同时将操作系统的反馈转换为用户能看懂的反馈,解决了用户和操作系统沟通成本的问题。与此同时,命令行解释器还能够拦…